【从零开始的rust web开发之路 四】rust语言tokio异步使用redis教程
文章目录
- 前言
- 一、首先引入依赖
- 二、创建redis客户端
- 三、相关操作
- 设置值
- mset设置多个key值
- 设置含有过期时间的值
- 如果key不存在才设置
- 获取基本类型值
- 删除一个键
- 删除多个键
- 判断键是否存在
- 如何使用json序列化
- 导入相关依赖
- 代码相关实例
- 总结
前言
使用rust写web,自然是离不开redis缓存的。rust也有现成redis连接库,并且支持阻塞和异步两种模式。下面教程我我以tokio异步操作redis方式为主,同步的操作可以参考异步,区别不大
一、首先引入依赖
redis = { version = "0.24.0",features = [ "r2d2" , "tokio-comp" ] }
我这里feature开启了r2d2连接池,以及开启了tokio异步支持特性。
全部特性列表如下
可选功能
定义了一些可以启用其他功能的功能 如果需要的话。其中一些是默认打开的。
- acl:启用 ACL 支持(默认启用)
- aio:启用异步 IO 支持(默认启用)
- geospatial:启用地理空间支持(默认启用)
- script:启用脚本支持(默认启用)
- r2d2:启用 R2D2 连接池支持(可选)
- ahash:启用AHASH地图/设置支持并在内部使用AHASH(+7-10%性能)- (可选)
- cluster:启用 Redis集群支持(可选)
- cluster-async:启用异步 Redis 集群支持(可选)
- tokio-comp:启用对 Tokio的支持(可选)
- connection-manager:启用对自动重新连接的支持(可选)
- keep-alive:通过板启用保持活动选项(可选)socket2
二、创建redis客户端
let client = redis::Client::open("redis://127.0.0.1:6379/").unwrap();//获取连接let con = client.get_async_connection().await.expect("连接redis失败");
此处url的格式为redis://[][:@][:port][/]
db是redis数据库索引。可根据业务自行修改
三、相关操作
设置值
#[tokio::main]
async fn main() {let client = redis::Client::open("redis://127.0.0.1:6379/10").unwrap();let mut con = client.get_async_connection().await.expect("连接redis失败");con.set::<&str,u32,()>("my_key", 12).await.expect("操作失败");
}
此处<&str,u32,()>前两个泛型指定key,value。第三个泛型是实现了FromRedisValue Trait的返回,set方法可以不指定
mset设置多个key值
#[tokio::main]
async fn main() {let client = redis::Client::open("redis://127.0.0.1:6379/10").unwrap();let mut con = client.get_async_connection().await.expect("连接redis失败");con.mset::<&str,u32,()>(&[("my_key1", 100), ("my_key2", 200)]).await.expect("操作失败");
}
设置含有过期时间的值
#[tokio::main]
async fn main() {let client = redis::Client::open("redis://127.0.0.1:6379/10").unwrap();let mut con = client.get_async_connection().await.expect("连接redis失败");//设置60秒过期,第三个参数是过期时间con.pset_ex::<&str,u32,()>("my_key", 12,60*1000).await.expect("操作失败");
}
pset_ex是毫秒单位,set_ex是秒
如果key不存在才设置
#[tokio::main]
async fn main() {let client = redis::Client::open("redis://127.0.0.1:6379/10").unwrap();let mut con = client.get_async_connection().await.expect("连接redis失败");con.set_nx::<&str,u32,()>("key3",30).await.expect("操作失败");
}
获取基本类型值
#[tokio::main]
async fn main() {let client = redis::Client::open("redis://127.0.0.1:6379/10").unwrap();let mut con = client.get_async_connection().await.expect("连接redis失败");let value : u32 = con.get("my_key").await.expect("获取值失败");println!("my_key = {}", value);
}
删除一个键
#[tokio::main]
async fn main() {let client = redis::Client::open("redis://127.0.0.1:6379/10").unwrap();let mut con = client.get_async_connection().await.expect("连接redis失败");con.del::<&str, ()>("key3") .await.expect("删除redis key失败");
}
删除多个键
#[tokio::main]
async fn main() {let client = redis::Client::open("redis://127.0.0.1:6379/10").unwrap();let mut con = client.get_async_connection().await.expect("连接redis失败");// 多个键删除let keys_to_delete = vec!["my_key1", "my_key2"];let result = con.del::<&Vec<&str>, i32>(&keys_to_delete).await;match result {Ok(count) => println!("Deleted {} keys", count),Err(e) => println!("Error: {}", e),}
}
判断键是否存在
#[tokio::main]
async fn main() {let client = redis::Client::open("redis://127.0.0.1:6379/10").unwrap();let mut con = client.get_async_connection().await.expect("连接redis失败");// 判断键是否存在let exists : bool = con.exists("my_key").await.expect("执行redis命令失败");println!("exists: {}", exists);
}
更多详细的接口文档可以看官网AsyncCommands接口https://docs.rs/redis/0.24.0/redis/trait.AsyncCommands.html
如何使用json序列化
可以通过自定义方法添加泛型约束实现功能。使用serde_json库来进行序列化
导入相关依赖
[dependencies]
tokio = { version = "1.35.1", features = ["full"] }
redis = { version = "0.24.0",features = [ "r2d2" , "tokio-comp","json" ] }
serde_json = "1.0.111"
serde = { version = "1.0.195", features = ["derive"] }
代码相关实例
use redis::{aio, AsyncCommands, RedisError, RedisResult, ToRedisArgs};
use serde::{Deserialize, Serialize};
use serde::de::DeserializeOwned;#[derive(Debug, Clone,Deserialize, Serialize)]
struct User{name: String,age: u8,email: String,
}#[tokio::main]
async fn main() {let client = redis::Client::open("redis://127.0.0.1:6379/10").unwrap();let mut con = client.get_async_connection().await.expect("连接redis失败");// 判断键是否存在let user = User{name: "zhangsan".to_string(),age: 18,email: "11111111".to_string(),};set_json(&mut con,"user1",&user).await.expect("设置失败");let user :User = get_json(&mut con, "user2").await.expect("获取失败,没有找到User");println!("user:{:?}",user);
}//设置json格式对象
pub async fn set_json<K,T>(con: &mut aio::Connection, key: K, obj : &T) -> RedisResult<String>where K: ToRedisArgs + Send + Sync, T: Serialize{let json = serde_json::to_string(obj);match json {Ok(obj) => {con.set::<K,String, String>(key,obj).await},Err(e) => {return Err(RedisError::from(e));}}
}//获得json格式对象
pub async fn get_json<K, T>(con: &mut aio::Connection, key: K) -> RedisResult<T>where K: ToRedisArgs + Send + Sync, T: DeserializeOwned{let json : String = con.get(key).await?;let result = serde_json::from_str(&json);return match result {Ok(obj) => { Ok(obj) },Err(e) => { Err(RedisError::from(e)) }}
}
通过这两个方法泛型约束据即可实现相关功能
总结
以上就是今天要讲的内容,本文介绍了rust语言tokio异步使用redis教程,后续会出一个扩展使用教程
相关文章:
【从零开始的rust web开发之路 四】rust语言tokio异步使用redis教程
文章目录 前言一、首先引入依赖二、创建redis客户端三、相关操作设置值mset设置多个key值设置含有过期时间的值如果key不存在才设置获取基本类型值删除一个键删除多个键判断键是否存在 如何使用json序列化导入相关依赖代码相关实例 总结 前言 使用rust写web,自然是…...
uniapp本地存储的几种方式localStorage
在uniapp开发中,本地存储是一个常见的需求。本地存储可以帮助我们在客户端保存和管理数据,以便在应用程序中进行持久化存储。本文将介绍uniapp中本地存储的几种方式,以及相关的代码示例。 介绍 在移动应用开发中,我们经常需要将…...
扩展学习|统计学习理论(SLT)与极限学习机(ELM)应用于大社会数据分析
文献来源:[1] Oneto L , Bisio F , Cambria E ,et al.Statistical Learning Theory and ELM for Big Social Data Analysis[J].IEEE Computational Intelligence Magazine, 2016, 11(3):45-55.DOI:10.1109/MCI.2016.2572540. 提取链接:链接:h…...
配置实例—交换机VLAN聚合配置实例
一、组网需求 某公司拥有多个部门且位于同一网段,为了提升业务安全性,将不同部门的用户划分到不同VLAN中。现由于业务需要,不同部门间的用户需要互通。如图1所示,VLAN2和VLAN3为不同部门,现需要实现不同VLAN间的用户可…...
网络开发的隐形壁垒:如何巧妙解决跨域难题?
什么是跨域 跨域是浏览器受同源(协议、域名、端口)策略的限制,不允许不同源的站点之间进行某些操作(如发送ajax请求,操作dom,读取cookie),如果不进行特殊配置是不能操作成功的&…...
【极简】conda同一个服务器上迁移环境 export / create
导出 直接看conda的document:https://docs.conda.io/projects/conda/en/latest/commands/env/export.html conda env export conda env export --file SOME_FILE重建 conda documentation: https://docs.conda.io/projects/conda/en/latest/commands/env/create.…...
HBase 数据导入导出
HBase 数据导入导出 1. 使用 Docker 部署 HBase2. HBase 命令查找3. 命令行操作 HBase3.1 HBase shell 命令3.2 查看命名空间3.3 查看命名空间下的表3.4 新建命名空间3.5 查看具体表结构3.6 创建表 4. HBase 数据导出、导入4.1 导出 HBase 中的某个表数据4.2 导入 HBase 中的某…...
(java版)排序算法----【冒泡,选择,插入,希尔,快速排序,归并排序,基数排序】超详细~~
目录 冒泡排序(BubbleSort): 代码详解: 冒泡排序的优化: 选择排序(SelectSort): 代码详解: 插入排序(InsertSort): 代码详解: 希尔排序(ShellSort): 法一…...
服务器托管的作用是什么?
服务器托管是将企业的服务器和相关设备托管到具有完善机房设施、高品质网络环境与运营经验的网络数据中心内,服务器托管在维护方面一般是由客户负责的,或者是由其他的授权人进行远程维护。 那服务器托管的作用都有哪些呢? 服务器托管不需要企…...
美团启动架构调整:聚力核心本地商业,提升科技与境外业务优先级
2月2日,美团CEO王兴发布内部邮件宣布新的组织架构调整。邮件显示,美团对核心本地商业相关多项业务进行了整合,并进一步提升了科技与国际化相关业务的优先级。 在核心本地商业上,美团对过去相对独立的事业群进行了整合。主要调整包…...
监测Tomcat项目宕机重启脚本(Linux)
1.准备好写好的脚本 #!/bin/sh # 获取tomcat的PID TOMCAT_PID$(ps -ef | grep tomcat | grep -v tomcatMonitor |grep -v grep | awk {print $2}) # tomcat的启动文件位置 START_TOMCAT/mnt/tomcat/bin/startup.sh # 需要监测的一个GET请求地址 MONITOR_URLhttp://localhost:…...
道可云元宇宙每日资讯|北京:推进元宇宙在智慧城市应用
道可云元宇宙每日简报(2024年2月2日)讯,今日元宇宙新鲜事有: 石狮市检察院“元宇宙智慧展馆”正式启用 为深入实施数字检察战略,主动探索元宇宙技术在未成年人检察、公益诉讼检察等方面的应用,打造集案件…...
Logback学习
logback 1、logback介绍 Logback是由log4j创始人设计的另一个开源日志组件,性能比log4j要好。 lockback优点: 内核重写、测试充分、初始化内存加载更小,这一切让logback性能和log4j相比有诸多倍的提升。logback非常自然地直接实现了slf4j…...
【Chrono Engine学习总结】2-可视化
由于Chrono的官方教程在一些细节方面解释的并不清楚,自己做了一些尝试,做学习总结。 0、基本概念 类型说明: Chrono的可视化包括两块:实时可视化,以及离线/后处理可视化。 其中,实时可视化,又…...
pytorch创建tensor
目录 1. 从numpy创建2. 从list创建3. 创建未初始化tensor4. 设置默认tensor创建类型5. rand/rand_like, randint6. randn生成正态分布随机数7. full8. arange/range9. linspace/logspace10. Ones/zeros/eye11. randperm 1. 从numpy创建 2. 从list创建 3. 创建未初始化tensor T…...
Cmake语法学习3:语法
1.双引号 1.1 命令参数 1)介绍 命令中多个参数之间使用空格进行分隔,而 cmake 会将双引号引起来的内容作为一个整体,当它当成一个参数,假如你的参数中有空格(空格是参数的一部分),那么就可以使…...
JavaScript 基础 - 第1天
介绍 掌握 JavaScript 的引入方式,初步认识 JavaScript 的作用 引入方式 JavaScript 程序不能独立运行,它需要被嵌入 HTML 中,然后浏览器才能执行 JavaScript 代码。通过 script 标签将 JavaScript 代码引入到 HTML 中,有两种方式…...
人口增长问题 T1063
#include<bits/stdc.h> using namespace std; int main(){int n;double x;cin>>x>>n;for(int i1;i<n;i){xx*1.001;}printf("%.4lf",x);return 0; }...
2024年Java算法面试题
2024年Java实战面试题(北京)_java 5 年 面试-CSDN博客 一、波菲那契递归 System.out.println("banc " banc(10)) public static int banc(int n){if( n0 ){return 0;}else if( n1 ){return 1;}else{return banc(n-1) banc(n-2);} } 二、冒…...
C#——三角形面积公式
已知三角形的三个边,求面积,可以使用海伦公式。 因此,可以执行得到三角形面积公式的计算方法代码如下: /** / <summary>* / 三角形面积公式* / </summary>* / <param name"a">边长a</param>*…...
观成科技:隐蔽隧道工具Ligolo-ng加密流量分析
1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具,该工具基于TUN接口实现其功能,利用反向TCP/TLS连接建立一条隐蔽的通信信道,支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式,适应复杂网…...
Java多线程实现之Callable接口深度解析
Java多线程实现之Callable接口深度解析 一、Callable接口概述1.1 接口定义1.2 与Runnable接口的对比1.3 Future接口与FutureTask类 二、Callable接口的基本使用方法2.1 传统方式实现Callable接口2.2 使用Lambda表达式简化Callable实现2.3 使用FutureTask类执行Callable任务 三、…...
MODBUS TCP转CANopen 技术赋能高效协同作业
在现代工业自动化领域,MODBUS TCP和CANopen两种通讯协议因其稳定性和高效性被广泛应用于各种设备和系统中。而随着科技的不断进步,这两种通讯协议也正在被逐步融合,形成了一种新型的通讯方式——开疆智能MODBUS TCP转CANopen网关KJ-TCPC-CANP…...
VTK如何让部分单位不可见
最近遇到一个需求,需要让一个vtkDataSet中的部分单元不可见,查阅了一些资料大概有以下几种方式 1.通过颜色映射表来进行,是最正规的做法 vtkNew<vtkLookupTable> lut; //值为0不显示,主要是最后一个参数,透明度…...
【Web 进阶篇】优雅的接口设计:统一响应、全局异常处理与参数校验
系列回顾: 在上一篇中,我们成功地为应用集成了数据库,并使用 Spring Data JPA 实现了基本的 CRUD API。我们的应用现在能“记忆”数据了!但是,如果你仔细审视那些 API,会发现它们还很“粗糙”:有…...
C# SqlSugar:依赖注入与仓储模式实践
C# SqlSugar:依赖注入与仓储模式实践 在 C# 的应用开发中,数据库操作是必不可少的环节。为了让数据访问层更加简洁、高效且易于维护,许多开发者会选择成熟的 ORM(对象关系映射)框架,SqlSugar 就是其中备受…...
聊一聊接口测试的意义有哪些?
目录 一、隔离性 & 早期测试 二、保障系统集成质量 三、验证业务逻辑的核心层 四、提升测试效率与覆盖度 五、系统稳定性的守护者 六、驱动团队协作与契约管理 七、性能与扩展性的前置评估 八、持续交付的核心支撑 接口测试的意义可以从四个维度展开,首…...
Android Bitmap治理全解析:从加载优化到泄漏防控的全生命周期管理
引言 Bitmap(位图)是Android应用内存占用的“头号杀手”。一张1080P(1920x1080)的图片以ARGB_8888格式加载时,内存占用高达8MB(192010804字节)。据统计,超过60%的应用OOM崩溃与Bitm…...
【Go语言基础【12】】指针:声明、取地址、解引用
文章目录 零、概述:指针 vs. 引用(类比其他语言)一、指针基础概念二、指针声明与初始化三、指针操作符1. &:取地址(拿到内存地址)2. *:解引用(拿到值) 四、空指针&am…...
人机融合智能 | “人智交互”跨学科新领域
本文系统地提出基于“以人为中心AI(HCAI)”理念的人-人工智能交互(人智交互)这一跨学科新领域及框架,定义人智交互领域的理念、基本理论和关键问题、方法、开发流程和参与团队等,阐述提出人智交互新领域的意义。然后,提出人智交互研究的三种新范式取向以及它们的意义。最后,总结…...
