当前位置: 首页 > news >正文

C++ 操作 Redis 客户端

引言

前面几篇文章都在介绍 Redis 原生命令行客户端,在实际应用开发中,开发人员更希望使用针对特定编程语言的专用客户端,通过编程的方式操作 Redis 数据库。因此,Redis 支持多种编程语言。本文将介绍 如何使用 C++ 语言来操作 Redis

一、认识 RESP 协议

RESP(Redis serialization protocol)是 Redis 自定义的应用层协议,所有编程语言的 Redis 库都是基于这个协议开发的。它设计的核心目标是简单易实现、解析高效,同时兼顾人类可读性,使得开发者能轻松调试 Redis 通信过程。这里仅做简单介绍,详细内容可参考Redis 官方文档 :Redis 的 RESP 协议

1. RESP 协议的核心特点

  1. 文本协议:基于 ASCII 字符编码,协议内容可直接通过肉眼识别(区别于二进制协议),便于调试。
  2. 类型明确:通过特定前缀字符标识数据类型(如字符串、整数、数组等),解析时无需额外判断。
  3. 高效解析:协议格式规则简单,解析器可通过线性扫描快速处理,几乎无性能开销。
  4. 扩展性强:支持复杂数据结构(如嵌套数组),能满足 Redis 多样化命令和返回值的需求。
  5. 一问一答式:客户端一次请求,服务器就会回答一次

2. RESP 协议的基本数据类型及格式

RESP 通过首字符区分数据类型,常见类型及格式如下:

类型前缀字符格式说明示例
简单字符串++字符串内容\r\n(不含换行符,用于返回简单结果,如 “OK”)+OK\r\n
错误消息--错误类型: 错误描述\r\n(用于返回错误信息,如命令不存在)-ERR unknown command 'foo'\r\n
整数::整数数值\r\n(用于返回计数、状态码等,如 EXISTS 命令的返回值):1\r\n(表示存在该键)
批量字符串$$长度\r\n内容\r\n(用于存储二进制安全的字符串,长度为 -1 表示空值)$5\r\nhello\r\n(表示字符串 “hello”)
数组**元素数量\r\n元素1\r\n元素2\r\n...(可嵌套数组,用于表示命令或复杂返回值)*3\r\n$3\r\nSET\r\n$5\r\nmykey\r\n$7\r\nmyvalue\r\n(表示 SET mykey myvalue 命令)

3.、Redis 命令的传输流程(基于 RESP)

Redis 客户端与服务器的通信均通过 RESP 协议完成,以 SET mykey "hello" 命令为例,流程如下:

  1. 客户端发送命令
    命令以数组类型传输,数组元素为命令名和参数(均为批量字符串):

    *3\r\n          // 数组包含3个元素(SET、mykey、hello)  
    $3\r\nSET\r\n   // 第一个元素:命令名 "SET"(长度3)  
    $5\r\nmykey\r\n // 第二个元素:键 "mykey"(长度5)  
    $5\r\nhello\r\n // 第三个元素:值 "hello"(长度5)  
    
  2. 服务器返回结果
    若命令执行成功,返回简单字符串 +OK\r\n;若失败,返回错误消息(如 -ERR syntax error\r\n)。

4. RESP 协议的优势与适用场景

  • 优势

    • 实现简单:开发者可快速编写解析器(无需处理复杂的二进制格式)。
    • 调试友好:通过 telnetnc 等工具可直接发送 RESP 格式命令与 Redis 交互(如 telnet localhost 6379 后输入 *2\r\n$4\r\nPING\r\n$0\r\n\r\n 测试连接)。
    • 兼容性强:支持所有 Redis 命令及返回值类型,包括复杂结构(如 HGETALL 返回的键值对数组)。
  • 适用场景

    • 所有 Redis 客户端与服务器的通信(如 C++ 的 redisplusplus、Python 的 redis-py、Java 的 Jedis 等均基于 RESP 实现)。
    • 自定义 Redis 客户端开发(需严格遵循 RESP 格式组装命令和解析响应)。

我们实际使用时,不需要按照协议自动组织,只需要使用Redis 官方提供或者大佬们开源的库就可以,我们接下来将使用 redis-plus-plus 库,它是 C++ 语言的 Redis库

二、ubuntu 安装 redis-plus-plus

1. 安装 hiredis

hiredis 是 Redis 官方为 c 语言提供的客户端支持,是redis-plus-plus 必备的依赖库
通过包管理器来安装:

sudo apt install libhiredis-dev

2.安装 redis-plus-plus

只能通过源码编译的方式安装

git clone https://github.com/sewenew/redis-plus-plus.gitcd redis-plus-plusmkdir buildcd buildcmake ..makesudo make installcd ..

依次执行完命令就安装好了,然后创建的这些目录我们都可以直接删除了

三、c++ 操作 Redis

环境安装好了,我们就可以通过编写 C++ 代码来操作 Redis 了

1. c++ redis的第一个程序

hello.cc

#include <sw/redis++/redis++.h>
#include <sw/redis++/redis.h>#include <iostream>
#include <string>
#include <unordered_map>
#include <vector>using std::cout;
using std::endl;
using std::string;
using std::unordered_map;
using std::vector;int main()
{// 创建 Redis 对象,需指定 Redis服务器ip地址和端口sw::redis::Redis redis("tcp://127.0.0.1:6379");// 调用 ping 方法,让客户端发送一个 ping 命令,服务端会返回一个 pongstd::string result = redis.ping();std::cout << result << std::endl;return 0;
}

这段代码就是创建一个 Redis 对象,然后向 Redis 服务器发送一个 ping 命令,于是 Redis 服务器就会返回一个 pong 。

Makefile编写
我们需要注意 makefile 编写中需要指定库的位置,否则无法编译

hello:hello.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadPHONY:clean
clean:rm -rf hello

编译运行:

至此,使用 c++ 操作 Redis 的第一个程序就完成了!

2. 使用 c++ 执行 Redis 通用命令

代码框架


#include <sw/redis++/redis++.h>
#include <sw/redis++/redis.h>#include <iostream>
#include <string>void test()
{// 每个命令写在这里// ...
}int main()
{sw::redis::Redis redis("tcp://127.0.0.1:6379");return 0;
}

首先给出代码框架,后续测试命令都会新创建一个 test 函数,然后再主函数中调用
Makefile:

.PHONY:all
all: hello generic	
hello:hello.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthread
generic:generic.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadPHONY:clean
clean:rm -rf hello generic

get 和 set

set接口

可以看到这些参数和set命令的选项有很大关联:

  • StringView 类型是 redis-plus-plus 提供的只读字符串,不能修改,这是由于只读字符产效率更高
  • 参数 key参数 val: 都是 StringView 类型
  • 参数 ttl:以毫秒为单位的过期时间
  • updatetype : 相当于 NX 和 XX

get接口

  • get的返回值是 OptionalString,这个类型会对 空值 (nil)做特殊处理,c++ 14后也引入了 std::optional_string,具体用法见示例代码。
// get 和 set
void test1(sw::redis::Redis &redis)
{std::cout << "get 和 set 的使用" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();// 使用 set 设置几个 keyredis.set("key1", "111");// 使用 get 获取到 key 对应的 valueauto value1 = redis.get("key1");if (value1){std::cout << "value1=" << value1.value() << std::endl;}// key2 是空值,这里如果不加判断会抛出异常auto value2 = redis.get("key2");if (value2){std::cout << "value2=" << value2.value() << std::endl;}
}

运行结果(不要忘记在主函数中调用exists):
value2是空值,由于if语句,不会输出

exists


返回类型是 long long,支持查询多个key,返回存在的个数
generic.cc

// exists 命令
void test2(sw::redis::Redis &redis)
{std::cout << "exists 的 使用" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();redis.set("key1", "value1");redis.set("key2", "value2");auto ret = redis.exists("key1");std::cout << ret << std::endl;ret = redis.exists("key3");std::cout << ret << std::endl;ret = redis.exists({"key1", "key2", "key3"});std::cout << ret << std::endl;
}

运行结果(记得在主函数中调用):

del

返回删除成功的个数

// del 命令
void test3(sw::redis::Redis &redis)
{std::cout << "del 的使用" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();redis.set("key1", "value1");redis.set("key2", "value2");auto ret = redis.del({"key1", "key2", "key3"});std::cout << ret << std::endl;ret = redis.exists({"key1", "key2", "key3"});std::cout << ret << std::endl;
}

运行结果:

keys

  • 参数 pattern : 匹配规则
  • 参数 output: 插入迭代器,我们可以事先构建一个容器,传入迭代器,这样 keys就会把查询到的元素插入容器中
// keys 命令
void test4(sw::redis::Redis &redis)
{std::cout << "keys 的使用" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();redis.set("key1", "value1");redis.set("key2", "value2");redis.set("key3", "value2");redis.set("key4", "value2");redis.set("key5", "value2");// redis 的keys 有两个参数,第一个是匹配规则// 第二个是 “插入迭代器”,我们可以事先构建一个容器,传入迭代器// 这样 keys就会把查询到的元素插入容器中std::vector<std::string> result;auto it = std::back_inserter(result);redis.keys("*", it);// 打印容器相关内容for (auto &str : result){std::cout << str << std::endl;}
}

expire 和 ttl

// expire and ttl
void test5(sw::redis::Redis &redis)
{std::cout << "expire and ttl" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();redis.set("key1", "111");redis.expire("key1", std::chrono::seconds(10));std::this_thread::sleep_for(std::chrono::seconds(5));long long ret = redis.ttl("key1");std::cout << "剩余时间: " << ret << std::endl;
}

type

返回值是string类型

// type
void test6(sw::redis::Redis &redis)
{std::cout << "type" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();redis.set("key1", "111");redis.lpush("key2", "111");std::cout << redis.type("key1") << std::endl;std::cout << redis.type("key2") << std::endl;
}

执行结果:

3. String

代码框架

#include <sw/redis++/cxx_utils.h>
#include <sw/redis++/redis++.h>
#include <sw/redis++/redis.h>#include <iostream>
#include <iterator>
#include <vector>// mget 和 mset
void test1(sw::redis::Redis& redis)
{
}int main()
{sw::redis::Redis redis("tcp://127.0.0.1:6379");// test1(redis);// test2(redis);test3(redis);return 0;
}

Makefile:

.PHONY:all
all: hello generic stringhello:hello.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadgeneric:generic.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadstring:string.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadPHONY:clean
clean:rm -rf hello generic string

mset 和 mget

支持一次性设置多个值和获取多个值

// mget 和 mset
void test1(sw::redis::Redis& redis)
{std::cout << "mget 和 mset" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();// 第一种方法,直接使用初始化列表// redis.mset({std::make_pair("key1", "1"), std::make_pair("key2", "2"),//             std::make_pair("key3", "3")});// 第二种方法,通过迭代器构造std::vector<std::pair<std::string, std::string>> keys = {{"key1", "111"}, {"key2", "222"}, {"key3", "333"}};redis.mset(keys.begin(), keys.end());std::vector<sw::redis::OptionalString> results;auto it = std::back_inserter(results);redis.mget({"key1", "key2", "key3"}, it);for (auto& str : results){if (str){std::cout << str.value() << std::endl;}}
}
  • mset 支持两种方式设置:
    • 一是直接通过初始化列表传入pair
    • 二是构造一个容器,传入迭代器
  • mget 返回值是 OptionalString 也是通过插入迭代器的方式进行实现

getrange 和 setrange

void test2(sw::redis::Redis& redis)
{std::cout << "mget 和 mset" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();redis.set("key", "abcdefghjk");std::string result = redis.getrange("key", 2, 5);std::cout << "result: " << result << std::endl;redis.setrange("key", 2, "xyz");result = redis.getrange("key", 0, -1);std::cout << "result: " << result << std::endl;
}

运行结果:

incr 和 decr

// incr decr
void test3(sw::redis::Redis& redis)
{std::cout << "incr 和 decr" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();redis.set("key1", "100");redis.set("key2", "100");redis.incr("key1");redis.decr("key2");std::cout << redis.get("key1").value() << std::endl;std::cout << redis.get("key2").value() << std::endl;
}

运行结果:

4. List

代码框架

#include <sw/redis++/redis++.h>
#include <sw/redis++/redis.h>#include <iostream>
#include <iterator>int main()
{sw::redis::Redis redis("tcp://127.0.0.1:6379");// test1(redis);// test2(redis);test3(redis);return 0;
}

Makefile:

.PHONY:all
all: hello generic string listhello:hello.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadgeneric:generic.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadstring:string.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadlist:list.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadPHONY:clean
clean:rm -rf hello generic string list

lpush 和 lrange

// lpush 和 lrange
void test1(sw::redis::Redis& redis)
{std::cout << "lpush and lrange" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();// 插入单个元素redis.lpush("key", "1");// 插入一组元素,基于初始化列表redis.lpush("key", {"2", "3", "4"});// 插入一组元素,基于迭代器std::vector<std::string> values = {"5", "6", "7"};redis.lpush("key", values.begin(), values.end());// lrange 获取结果,需要构造容器,传入插入迭代器std::vector<std::string> results;auto it = std::back_inserter(results);redis.lrange("key", 0, -1, it);// 打印结果for (auto& result : results){std::cout << result << std::endl;}
}

lpop 和 rpop

// lpop 和 rpop
void test2(sw::redis::Redis& redis)
{std::cout << "lpop and rpop" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();redis.rpush("key", {"1", "2", "3", "4", "5", "6", "7"});auto ret = redis.lpop("key");if (ret){std::cout << "lpop: " << ret.value() << std::endl;}ret = redis.rpop("key");if (ret){std::cout << "rpop: " << ret.value() << std::endl;}
}

// blpop
void test3(sw::redis::Redis& redis)
{std::cout << "blpop" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();auto ret = redis.blpop({"key1", "key2", "key3"});if (ret){std::cout << "key: " << ret.value().first << std::endl;std::cout << "elem: " << ret.value().second << std::endl;}else{std::cout << "元素无效!" << std::endl;}
}


一开始会阻塞住,直到我们往 key1 key2 key3之一插入一个值

5. Set

代码框架

.PHONY:all
all: hello generic string list sethello:hello.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadgeneric:generic.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadstring:string.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadlist:list.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadset:set.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadPHONY:clean
clean:rm -rf hello generic string list set

sadd 和 smembers

// sadd smembers
void test1(sw::redis::Redis &redis)
{std::cout << "sadd and smembers" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();// 一次添加一个元素redis.sadd("key", "111");// 一次添加多个元素(使用初始化列表)redis.sadd("key", {"222", "333"});// 一次添加多个元素(使用迭代器)std::vector<std::string> elems = {"444", "555"};redis.sadd("key", elems.begin(), elems.end());// 通过 smembers 获取返回结果//  1. vector//  2. set// std::vector<std::string> results;// auto it = std::back_inserter(results);std::set<std::string> results;auto it = std::inserter(results, results.end());redis.smembers("key", it);// 打印结果for (auto &res : results){std::cout << res << std::endl;}
}

sismember 和 spop

// sismember and spop
void test2(sw::redis::Redis &redis)
{std::cout << "sismember and spop" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();redis.sadd("key", {"111", "222", "333"});bool result = redis.sismember("key", "111");std::cout << result << std::endl;redis.spop("key");redis.spop("key");redis.spop("key");result = redis.sismember("key", "111");std::cout << result << std::endl;
}

运行结果:

sinter 和 sinterstore

// sinter
void test3(sw::redis::Redis &redis)
{std::cout << "sinter" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();redis.sadd("key1", {"111", "222", "333"});redis.sadd("key2", {"111", "222", "444"});std::set<std::string> results;auto it = std::inserter(results, results.end());redis.sinter({"key1", "key2"}, it);for (auto &result : results){std::cout << result << std::endl;}
}// sinterstore
void test4(sw::redis::Redis &redis)
{std::cout << "sinterstore" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();redis.sadd("key1", {"111", "222", "333"});redis.sadd("key2", {"111", "222", "444"});long long len = redis.sinterstore("key3", {"key1", "key2"});std::cout << "len: " << len << std::endl;std::set<std::string> results;auto it = std::inserter(results, results.end());redis.smembers("key3", it);// 打印结果for (auto &res : results){std::cout << res << std::endl;}
}

6. Hash

代码框架

.PHONY:all
all: hello generic string list set hashhello:hello.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadgeneric:generic.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadstring:string.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadlist:list.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadset:set.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadhash:hash.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadPHONY:clean
clean:rm -rf hello generic string list set hash

hset 和 hget

// hget and hset
void test1(sw::redis::Redis &redis)
{std::cout << "hget and hset" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();// 插入一个 field-value 对redis.hset("key", "f1", "1");// 插入多个 field-value 对(初始化列表)redis.hset("key", {std::make_pair("f2", "2"), std::make_pair("f3", "3")});// 插入多个 field-value 对(迭代器)std::vector<std::pair<std::string, std::string>> fields = {std::make_pair("f4", "4"), std::make_pair("f5", "5")};auto result = redis.hget("key", "f1");if (result){std::cout << result.value() << std::endl;}else{std::cout << "返回值无效!" << std::endl;}
}

hexists,hdel 和 hlen

// hexists, hlen, hdel
void test2(sw::redis::Redis &redis)
{std::cout << "hget and hset" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();// 插入多个 field-value 对(初始化列表)redis.hset("key", {std::make_pair("f2", "2"), std::make_pair("f3", "3")});std::cout << "删除前:" << std::endl;long long len = redis.hlen("key");std::cout << "len: " << len << std::endl;bool result = redis.hexists("key", "f2");std::cout << result << std::endl;redis.hdel("key", "f2");std::cout << "删除后:" << std::endl;len = redis.hlen("key");std::cout << "len: " << len << std::endl;result = redis.hexists("key", "f2");std::cout << result << std::endl;
}

hkeys 和 hvals

// hkeys
// hkeys hvals
void test3(sw::redis::Redis &redis)
{std::cout << "keys hvals" << std::endl;// 清空一下数据库,避免之前数据的干扰,生产环境禁用!redis.flushall();// 插入多个 field-value 对(初始化列表)redis.hset("key", {std::make_pair("f1", "1"), std::make_pair("f2", "2"),std::make_pair("f3", "3")});std::vector<std::string> fields;auto it = std::back_inserter(fields);redis.hkeys("key", it);for (auto &str : fields){std::cout << str << std::endl;}std::vector<std::string> vals;it = std::back_inserter(vals);redis.hvals("key", it);for (auto &str : vals){std::cout << str << std::endl;}
}

7. Zset

.PHONY:all
all: hello generic string list set hash zsethello:hello.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadgeneric:generic.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadstring:string.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadlist:list.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadset:set.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadhash:hash.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadzset:zset.ccg++ -std=c++17 -o $@ $^ /usr/local/lib/libredis++.a /usr/lib/x86_64-linux-gnu/libhiredis.a -pthreadPHONY:clean
clean:rm -rf hello generic string list set hash zset

zadd 和 zrange

zrange 查询是否带分数取决于插入迭代器对应容器的类型:

  • 如果插入迭代器对应一个仅包含 string 的类型的容器
  • 如果插入迭代器对应容器是 pair<string, double> 为元素的容器,则返回分数
// zadd zrange
void test1(sw::redis::Redis& redis)
{std::cout << "zadd zrange" << std::endl;// 清空数据库,防止之前的数据干扰,生产环境禁用!redis.flushall();// 三种版本redis.zadd("key", "吕布", 99);redis.zadd("key", {std::make_pair("赵云", 98), std::make_pair("张飞", 95)});std::vector<std::pair<std::string, double>> members = {std::make_pair("典韦", 97), std::make_pair("关羽", 96)};redis.zadd("key", members.begin(), members.end());// zrange 是否查询分数要看容器类型//  如果容器仅 string 则 仅查询member//  如果容器是 pair<string, double>, 则返回member 和 分数std::cout << "仅查询member" << std::endl;std::vector<std::string> results;auto it = std::back_inserter(results);redis.zrange("key", 0, -1, it);for (auto& s : results){std::cout << s << std::endl;}std::cout << "查询member和分数" << std::endl;std::vector<std::pair<std::string, double>> scores;auto it1 = std::back_inserter(scores);redis.zrange("key", 0, -1, it1);for (auto& [member, score] : scores){std::cout << member << " " << score << std::endl;}
}

zcard zrank zscore

// zcard zscore zrank
void test2(sw::redis::Redis& redis)
{std::cout << "zcard zscore zrank" << std::endl;// 清空数据库,防止之前的数据干扰,生产环境禁用!redis.flushall();redis.zadd("key", "zhangsan", 90);redis.zadd("key", "lisi", 91);redis.zadd("key", "wangwu", 92);redis.zadd("key", "zhaoliu", 93);long long result = redis.zcard("key");std::cout << result << std::endl;auto score = redis.zscore("key", "zhangsan");if (score){std::cout << "score: " << score.value() << std::endl;}else{std::cout << "score 无效" << std::endl;}auto rank = redis.zrank("key", "lisi");if (score){std::cout << "rank: " << rank.value() << std::endl;}else{std::cout << "rank 无效" << std::endl;}
}

http://www.lryc.cn/news/612049.html

相关文章:

  • Docker swarm 常用的命令集合
  • Pipeline功能实现Redis批处理(项目批量查询点赞情况的应用)
  • 数据结构——双向链表及makefile
  • NineData 新增支持 AWS ElastiCache 复制链路
  • windows搬运文件脚本
  • 互斥锁与条件变量
  • 自然语言处理基础—(1)
  • 深入理解 C++ 中的stdpriority_queue:从原理到实战的高效优先级管理
  • ssm复习
  • 【嵌入式电机控制#26】BLDC:三相模拟采集
  • springboot项目前后端通用下载方法、问题和解决方案
  • PyTorch生成式人工智能(26)——使用PyTorch构建GPT模型
  • AVDTP Media Packet 传输全流程解析:从 SDP 到连接终止
  • 基于AntDesign二次封装table组件
  • 思途AOP学习笔记 0806
  • nginx代理出https,request.getRequestURL()得到http问题解决
  • 界面规范的其他框架实现-列表-layui实现
  • TypeError: crypto$2.getRandomValues is not a function
  • 北大、蚂蚁三个维度解构高效隐私保护机器学习:前沿进展+发展方向
  • AlexNet训练和测试CIFAR10
  • Python金融分析:从基础到量化交易的完整指南
  • 如何定位一个高并发场景下API响应时间从200ms突增到2s的问题
  • 生成模型实战 | Transformer详解与实现
  • 需求链路追踪
  • centos配置java环境变量
  • SpringCloud -- elasticsearch(二)
  • MonoFusion 与 Genie 3
  • 如何快速掌握大数据技术?大四学生用Spark和Python构建直肠癌数据分析与可视化系统
  • Apollo中三种相机外参的可视化分析
  • 「iOS」————单例与代理