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

axum 操作 redis

通过 redis-rs 这个 crate,可以很方便的操作 redis。它提供了同步和异步两种连接,由于我们要集成到 axum 中,所以这里使用异步连接。本章将展示如何获取 redis 异步连接、如何将字符串保存到 redis、如何获取到保存在 redis 里的字符串以及如何通过 redis 保存和读取自定义结构体。

获取 redis 异步连接

redis 默认端口是 6379,由于作者使用 docker 运行了多个 redis 实例,所以示例代码中可能不是redis默认端口。请根据你自身的环境将连接字符串中的主机和端口等信息进行修改。

redis 默认端口是 6379,由于作者使用 docker 运行了多个 redis 实例,所以示例代码中可能不是redis默认端口。请根据你自身的环境将连接字符串中的主机和端口等信息进行修改。

通过 redis::Client::open()方法可以建立与 redis 服务器的连接,然后使用get_async_connection()方法获取到异步连接。

let client = Client::open("redis://127.0.0.1:16379/").unwrap();
    let  conn = client
        .get_async_connection()
        .await
        .unwrap();
  • redis://[<用户名>][:<密码>@]<主机名>[:端口][/数据库]

  • unix:///<路径>[?db=<数据库>][&pass=<密码>][&user=<用户名>]

  • redis+unix:///<路径>[?db=<数据库>][&pass=<密码>][&user=<用户名>]

redis://[<用户名>][:<密码>@]<主机名>[:端口][/数据库]

unix:///<路径>[?db=<数据库>][&pass=<密码>][&user=<用户名>]

redis+unix:///<路径>[?db=<数据库>][&pass=<密码>][&user=<用户名>]

将字符串保存到 redis

调用异步连接的set()方法,可以将字符串保存到 redis:

conn.set("键名","值").await;

示例代码如下:

async fn set() -> Result<&'static str, String> {
    let client = Client::open(REDIS_DSN).map_err(|err| err.to_string())?;
    let mut conn = client
        .get_async_connection()
        .await
        .map_err(|err| err.to_string())?;
    conn.set("author", "axum.rs")
        .await
        .map_err(|err| err.to_string())?;
    Ok("Successfully set")
}

读取保存在 redis 中的字符串

异步连接的get()方法,用于从 redis 中获取指定键的数据:

let value = conn.get("键名").await;

示例代码如下:

async fn get() -> Result<String, String> {
    let client = Client::open(REDIS_DSN).map_err(|err| err.to_string())?;
    let mut conn = client
        .get_async_connection()
        .await
        .map_err(|err| err.to_string())?;
    let value = conn.get("author").await.map_err(|err| err.to_string())?;
    Ok(value)
}

由于 redis 保存的是底层的数据,所以你可以根据需要将读取到的数据进行类型转换,官方文档有示例。

由于 redis 保存的是底层的数据,所以你可以根据需要将读取到的数据进行类型转换,官方文档有示例。

自定义结构体和 redis

了解完字符串的读写操作,我们继续讨论在 redis 读写自定义结构体。

#[derive(Serialize, Deserialize)]
pub struct UserInfo {
    pub id: i32,
    pub username: String,
    pub email: String,
}

将自定义结构体写入 redis 服务器

我们需要将结构体序列化为字符串,然后再写入 redis 服务器:

从 redis 中读取自定义结构体

我们需要将 redis 中读取到的字符串反序列化为自定义结构体:

async fn get_user() -> Result<Json<UserInfo>, String> {
    let client = Client::open(REDIS_DSN).map_err(|err| err.to_string())?;
    let mut conn = client
        .get_async_connection()
        .await
        .map_err(|err| err.to_string())?;
    let value: String = conn.get("user").await.map_err(|err| err.to_string())?;
    let user: UserInfo = from_str(&value).map_err(|err| err.to_string())?;
    Ok(Json(user))
}

redis 支持在写入值的时候设置过期时间,时间一到,该数据自动删除:

conn.set_ex("键名","值",过期时间的秒数).await;

本章讨论了如何在 axum 集成 redis,完整代码可以在代码库中找到。

思考题

  • 本章展示的所有 handler 都有重复代码:获取 redis 连接,尝试将这段重复代码提取为独立的函数。

  • 更进一步,将写入和读取的操作封装成单独的函数。

  • 试试写入一个 30 秒过期的数据。

更进一步,将写入和读取的操作封装成单独的函数。

试试写入一个 30 秒过期的数据。

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