域名 AXUM.RS 将于2025年10月到期。我们无意再对其进行续费,我们希望你能够接续这个域名,让更多 AXUM 开发者继续受益。
  • 方案1️⃣AXUM.RS 域名 = 3000
  • 方案2️⃣方案1️⃣ + 本站所有专题原始 Markdown 文档 = 5000
  • 方案3️⃣方案2️⃣ + 本站原始数据库 = 5500
如果你有意接续这份 AXUM 情怀,请与我们取得联系。
说明:
  1. 如果有人购买 AXUM.RS 域名(方案1️⃣),或者该域名到期,本站将启用新的免费域名继续提供服务。
  2. 如果有人购买了 AXUM.RS 域名,且同时购买了内容和/或数据库(方案2️⃣/方案3️⃣),本站将关闭。届时我们或许会以另一种方式与你再相遇。

短链接跳转到原始链接

handler

/// 跳转到目标URL
pub async fn goto_url(
    Extension(state): Extension<AppState>,
    Path(id): Path<String>,
) -> HandlerRedirectResult {
    let handler_name = "goto_url";
    let client = get_client(&state, handler_name).await?;
    let result = db::goto_url(&client, id)
        .await
        .map_err(log_error(handler_name.to_string()))?;
    Ok(redirect(result.url.as_str()))
}

数据库

pub async fn goto_url(client: &Client, id: String) -> Result<UrlTarget> {
    let result = super::query_one(
        client,
        "UPDATE url SET visit=visit+1 WHERE id=$1 RETURNING url",
        &[&id],
    )
    .await?;
    Ok(result)
}

为什么不用事务

利益于 PostgreSQL 数据库的强大,我们可以使用单条 SQL 语句就实现既增加访问量,又返回原始链接。如果是其它数据库,则需要使用事务来分步操作:

// 1. 执行 UPDATE url SET visit=visit+1 WHERE id=?
// 2. 判断该 UPDATE 的结果,如果出错回滚事务
// 3. 执行 SELECT url FROM url WHERE id=?
// 4. 提交事务
// 5. 返回 SELECT 的结果
要查看完整内容,请先登录