域名 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️⃣),本站将关闭。届时我们或许会以另一种方式与你再相遇。

sqlx优雅地实现IN查询

在日常开发中,IN 查询是非常常见的需求,你会怎么来处理呢?借助 sqlx 的QueryBuilder 可以方便地实现。

假设有一个需求,需要返回一系列指定ID的会员信息的列表。对应的 SQL 语句大概这样:

注意,IN(???) 条件不是固定的,例中 IN(1,2,3,4) 只是举例,以便理解需求。

错误的实现

有其它语言开发经验的人肯定想到:

// 伪代码

let ids_str:Vec<String> = ids.iter().map(|id|id.to_string()).collect(); 
let ids_arg:String = ids_str.join(",");
let list:Vec<Member> = sqlx::query_as("SELECT * FROM member WHERE id IN(?)").bind(&ids_arg).fetch_all(conn).await.unwrap();

这种方式不会报错,甚至能工作的很好。但也可能遇到未知的BUG。

pub async fn select_in(conn: &sqlx::MySqlPool, ids: &[u32]) -> Result<Vec<model::member::Member>> {
    let mut q = sqlx::QueryBuilder::new("SELECT * FROM member WHERE id IN");
    q.push_tuples(ids.iter(), |mut b, id| {
        b.push_bind(id);
    });

    let ms = q
        .build_query_as()
        .fetch_all(conn)
        .await
        .map_err(Error::from)?;

    Ok(ms)
}

本章代码位于07/in分支。

要查看完整内容,请先登录