redis系列:通过文章点赞排名案例学习sortedset命令
前言
这一篇文章将讲述Redis中的sortedset类型命令,同样也是通过demo来讲述,其他部分这里就不在赘述了。
项目Github地址:https://github.com/rainbowda/learnWay/tree/master/learnRedis/case-sortedset
案例
demo功能是文章点赞排名等等,整个demo的大致页面如下。

准备工作
首先定义一个存储文章的key
private static final String ZSET_KEY = "articleList";
redis操作对象
private RedisTemplate redisTemplate;
//string 命令操作对象
private ValueOperations valueOperations;
//zset 命令操作对象
private ZSetOperations zSetOperations;
sortedset在Redis中的结构可以看下图(图片来源于Redis in Action)。

列表查询
@RequestMapping(value = "/getList/{sortType}", method = RequestMethod.GET)
public Set getList(@PathVariable String sortType) {
//如果没有数据,则添加10条数据
if (zSetOperations.size(ZSET_KEY) == 0){
for (int i = 1; i <= 10; i++) {
zSetOperations.add(ZSET_KEY,"文章:"+i, (int)(Math.random()*10+i));
}
}
//ASC根据分数从小到大排序,DESC反之
if ("ASC".equals(sortType)){
return zSetOperations.rangeWithScores(ZSET_KEY, 0, -1);
} else {
return zSetOperations.reverseRangeWithScores(ZSET_KEY, 0, -1);
}
}
这里为了省去一个个添加数据的麻烦,就在获取列表数据中加了个判断。当文章数据为0时,默认添加10条数据,设置随机score加上所在的索引。
然后根据url中的参数sortType来决定返回的数据是按照分数升序还是降序排序。功能效果如下

命令介绍
| 命令 | 用例 | 描述 |
|---|---|---|
| ZADD | ZADD key [NX|XX][CH] [INCR] score member [score member ...] | 将所有指定成员添加到键为key有序集合(sorted set)里面 |
| ZRANGE | ZRANGE key start stop [WITHSCORES] | 返回有序集key中,指定区间内的成员。其中成员的位置按score值递减(从小到大)来排列。 |
| ZREVRANGE | ZREVRANGE key start stop [WITHSCORES] | 返回有序集key中,指定区间内的成员。其中成员的位置按score值递减(从大到小)来排列。 |
赞或踩
java代码如下
@RequestMapping(value = "/star", method = RequestMethod.POST)
public boolean starOrUnStar(String member, String type) {
if ("UP".equals(type)){
zSetOperations.incrementScore(ZSET_KEY, member, 1);
} else {
zSetOperations.incrementScore(ZSET_KEY, member, -1);
}
return true;
}
根据type决定是否加减分数,当type为UP时表示赞,为其他(DOWN)时表示踩。功能效果如下

命令介绍
| 命令 | 用例 | 描述 |
|---|---|---|
| ZINCRBY | ZINCRBY key increment member | 为有序集key的成员member的score值加上增量increment |
升降序排名
java代码如下
@RequestMapping(value = "/rank/{type}/{member}", method = RequestMethod.GET)
public Long rank(@PathVariable String member, @PathVariable String type) {
Long rank = null;
if ("ASC".equals(type)){
rank = zSetOperations.rank(ZSET_KEY, member);
} else {
rank = zSetOperations.reverseRank(ZSET_KEY, member);
}
return rank;
}
根据type决定是升序排名还是降序排名,如果是ASC则调用rank方法获取升序排名,其他则调用reverseRank获取降序排名。与下方redis命令类似
ZRANK articleList "文章1"
ZREVRANK articleList "文章1"
页面效果图如下

命令介绍
| 命令 | 用例 | 描述 |
|---|---|---|
| ZRANK | ZRANK key member | 返回有序集key中成员member的排名。其中有序集成员按score值递增(从小到大)顺序排列。排名以0为底,也就是说,score值最小的成员排名为0。 |
| ZREVRANK | ZREVRANK key member | 返回有序集key中成员member的排名,其中有序集成员按score值从大到小排列。 |
其他命令
获取属性
| 命令 | 用例 | 描述 |
|---|---|---|
| ZCARD | ZCARD key | 返回key的有序集元素个数。 |
| ZCOUNT | ZCOUNT key min max | 返回有序集key中,score值在min和max之间(默认包括score值等于min或max)的成员个数。 |
| ZLEXCOUNT | ZLEXCOUNT key min max | 用于计算有序集合中指定成员之间的成员数量。 |
| ZSCORE | ZSCORE key member | 返回有序集key中,成员member的score值。 |
ZCARD命令
返回key的有序集元素个数。
ZCARD key
返回值:key存在的时候,返回有序集的元素个数,否则返回0。
redis客户端执行的命令如下
zadd zCardKey 1 one
zcard zCardKey
下面是java代码
@Test
public void zCard() {
jedis.zadd("zCardKey",1, "one");
jedis.zadd("zCardKey",2, "two");
System.out.println(jedis.zcard("zCardKey"));
System.out.println(zSetOperations.size("zCardKey"));
}
ZCOUNT命令
返回有序集key中,score值在min和max之间(默认包括score值等于min或max)的成员数量。
ZCOUNT key min max
返回值:指定分数范围的元素个数。
redis客户端执行的命令如下
zadd zCountKey 1 one 2 two 3 three 4 four
zcount zCountKey 2 3
执行结果如下

下面是java代码
@Test
public void zCount() {
jedis.zadd("zCountKey",1, "one");
jedis.zadd("zCountKey",2, "two");
jedis.zadd("zCountKey",3, "three");
jedis.zadd("zCountKey",4, "four");
System.out.println(jedis.zcount("zCountKey",2, 3));
System.out.println(zSetOperations.count("zCountKey",2, 3));
}
ZLEXCOUNT命令
计算有序集合中指定成员之间的成员数量(按成员字典正序排序),可以使用 - 和 + 表示score最小值和最大值
ZLEXCOUNT key min max
redis客户端执行的命令如下
ZADD zLexCountKey 2 "b" 1 "a" 3 "c" 5 "e" 4 "d"
ZLEXCOUNT zLexCountKey - +
ZLEXCOUNT zLexCountKey [b [d
执行结果如下

下面是java代码
@Test
public void zLexCount() {
zSetOperations.add("zLexCountKey", "b", 2);
zSetOperations.add("zLexCountKey", "a", 1);
zSetOperations.add("zLexCountKey", "c", 3);
zSetOperations.add("zLexCountKey", "e", 5);
zSetOperations.add("zLexCountKey", "d", 4);
System.out.println(jedis.zlexcount("zLexCountKey", "-", "+"));
System.out.println(jedis.zlexcount("zLexCountKey", "[b", "[d"));
}
ZSCORE命令
返回有序集key中,成员member的score值。
ZSCORE key member
返回值:成员member的score值
redis客户端执行的命令如下
zadd zScoreKey 1 one
ZSCORE zScoreKey one
下面是java代码
@Test
public void zScore() {
jedis.zadd("zScoreKey",1, "one");
System.out.println(jedis.zscore("zScoreKey", "one"));
System.out.println(zSetOperations.score("zScoreKey", "one"));
}
获取成员
| 命令 | 用例 | 描述 |
|---|---|---|
| ZRANGEBYLEX | ZRANGEBYLEX key min max [LIMIT offset count] |
返回指定成员区间内的成员,按成员字典正序排序。 |
| ZRANGEBYSCORE | ZRANGEBYSCORE key min max [WITHSCORES]``[LIMIT offset count] |
返回所有符合score条件的成员 |
| ZREVRANGEBYLEX | ZREVRANGEBYLEX key max min [LIMIT offset count] |
返回指定成员区间内的成员,按成员字典倒序排序。 |
| ZREVRANGEBYSCORE | ZREVRANGEBYSCORE key max min [WITHSCORES]``[LIMIT offset count] |
返回有序集合中指定分数区间内的成员,分数由高到低排序。 |
| ZSCAN | ZSCAN key cursor [MATCH pattern]``[COUNT count] |
请参考 SCAN |
ZRANGEBYLEX命令
返回指定成员区间内的成员,按成员字典正序排序。https://redis.io/commands/zrangebylex
ZRANGEBYLEX key min max [LIMIT offset count]
返回值:指定成员范围的元素列表。
redis客户端执行的命令如下
ZADD zRangeByLexKey 0 ba 0 a 0 ab 0 aa 0 b
ZRANGEBYLEX zRangeByLexKey - +
ZRANGEBYLEX zRangeByLexKey [aa (ba
执行结果如下

下面是java代码
@Test
public void zRangeByLex() {
zSetOperations.add("zRangeByLexKey", "ba", 0);
zSetOperations.add("zRangeByLexKey", "a", 0);
zSetOperations.add("zRangeByLexKey", "ab", 0);
zSetOperations.add("zRangeByLexKey", "aa", 0);
zSetOperations.add("zRangeByLexKey", "b", 0);
System.out.println(jedis.zrangeByLex("zRangeByLexKey", "-", "+"));
RedisZSetCommands.Range range = new RedisZSetCommands.Range();
range.gte("aa");
range.lt("ba");
System.out.println(zSetOperations.rangeByLex("zRangeByLexKey",range));
}
ZRANGEBYSCORE命令
获取score在范围之内的数据。min和max可以是-inf和+inf
ZRANGEBYSCORE key min max[WITHSCORES]``[LIMIT offset count]
redis客户端执行的命令如下
ZADD zRangeByScoreKey 1 ba 2 a 3 ab 4 aa 5 b
ZRANGEBYSCORE zRangeByScoreKey -inf +inf
ZRANGEBYSCORE zRangeByScoreKey 2 4
执行结果如下

下面是java代码
@Test
public void zRangeByScore() {
zSetOperations.add("zRangeByScoreKey", "ba", 1);
zSetOperations.add("zRangeByScoreKey", "a", 2);
zSetOperations.add("zRangeByScoreKey", "ab", 3);
zSetOperations.add("zRangeByScoreKey", "aa", 4);
zSetOperations.add("zRangeByScoreKey", "b", 5);
System.out.println(jedis.zrangeByScore("zRangeByScoreKey", "-inf", "+inf"));
RedisZSetCommands.Range range = new RedisZSetCommands.Range();
System.out.println(zSetOperations.rangeByScore("zRangeByScoreKey", 2, 4));
}
移除相关命令
| 命令 | 用例 | 描述 |
|---|---|---|
| ZREM | ZREM key member [member ...] | 删除有序集合中的成员 |
| ZREMRANGEBYLEX | ZREMRANGEBYLEX key min max | 删除名称按字典由低到高排序成员之间所有成员 |
| ZREMRANGEBYRANK | ZREMRANGEBYRANK key start stop | 移除有序集key中,指定排名(rank)区间内的所有成员。 |
| ZREMRANGEBYSCORE | ZREMRANGEBYSCORE key min max | 移除有序集key中,所有score值介于min和max之间(包括等于min或max)的成员 |
ZREM命令
ZREM key member [member ...]
返回值:有序集合中删除的成员个数
redis客户端执行的命令如下
ZADD zRemKey 1 "one" 2 "two" 3 "three"
ZREM zRemKey one
ZRANGE zRemKey 0 -1
执行结果如下

下面是java代码
@Test
public void zRem() {
zSetOperations.add("zRemKey", "one", 1);
zSetOperations.add("zRemKey", "two", 2);
zSetOperations.add("zRemKey", "three", 3);
//jedis.zrem("zRemKey", "one");
zSetOperations.remove("zRemKey", "one");
System.out.println(zSetOperations.range("zRemKey", 0 , -1));
}
交并集
| 命令 | 用例 | 描述 |
|---|---|---|
| ZINTERSTORE | ZINTERSTORE destination numkeys key [key ...]``[WEIGHTS weight] [SUM\|MIN\|MAX] |
计算给定的numkeys个有序集合的交集,并且把结果放到destination中 |
| ZUNIONSTORE | ZUNIONSTORE destination numkeys key [key ...]``[WEIGHTS weight] [SUM\|MIN\|MAX] |
计算给定的numkeys个有序集合的并集,并且把结果放到destination中 |
ZINTERSTORE命令
计算给定的numkeys个有序集合的交集,并且把结果放到destination中。
在给定要计算的key和其它参数之前,必须先给定key个数(numberkeys)。
默认情况下,结果中一个元素的分数是有序集合中该元素分数之和,前提是该元素在这些有序集合中都存在。因为交集要求其成员必须是给定的每个有序集合中的成员,结果集中的每个元素的分数和输入的有序集合个数相等。
对于WEIGHTS和AGGREGATE参数的描述,参见命令ZUNIONSTORE。
如果destination存在,就把它覆盖。
ZINTERSTORE destination numkeys key [key ...] [WEIGHTS weight [weight ...]] [AGGREGATE SUM|MIN|MAX]
返回值:结果有序集合destination中元素个数。
redis客户端执行的命令如下
ZADD zInterStoreKey1 1 "one" 2 "two"
ZADD zInterStoreKey2 1 "one" 2 "two" 3 "three"
ZINTERSTORE zInterStoreSumResult 2 zInterStoreKey1 zInterStoreKey2 WEIGHTS 2 3
ZRANGE zInterStoreSumResult 0 -1 WITHSCORES
执行结果如下

下面是java代码
@Test
public void zInterStore() {
zSetOperations.add("zInterStoreKey1", "one", 1);
zSetOperations.add("zInterStoreKey1", "two", 2);
zSetOperations.add("zInterStoreKey2", "one", 1);
zSetOperations.add("zInterStoreKey2", "two", 2);
zSetOperations.add("zInterStoreKey2", "three", 3);
ZParams zParams = new ZParams();
zParams.weightsByDouble(2, 3);
zParams.aggregate(ZParams.Aggregate.SUM);
jedis.zinterstore("zInterStoreSumResult", zParams, "zInterStoreKey1", "zInterStoreKey2");
printTuple("zInterStoreSumResult", jedis.zrangeWithScores("zInterStoreSumResult", 0, -1));
}
ZUNIONSTORE命令
计算给定的numkeys个有序集合的并集,并且把结果放到destination中。
WEIGHTS参数相当于权重,默认就是1,可以给不同的key设置不同的权重
AGGREGATE参数默认使用的参数SUM,还可以选择MIN或者MAX。这个参数决定结果集的score是取给定集合中的相加值、最小值还是最大值
ZUNIONSTORE destination numkeys key [key ...] [WEIGHTS weight [weight ...] ] [AGGREGATE SUM|MIN|MAX]
redis客户端执行的命令如下
ZADD zUnionStoreKey1 1 "one" 2 "two"
ZADD zUnionStoreKey2 1 "one" 2 "two" 3 "three"
ZUNIONSTORE zUnionStoreSumResult 2 zUnionStoreKey1 zUnionStoreKey2 WEIGHTS 2 3
ZUNIONSTORE zUnionStoreMinResult 2 zUnionStoreKey1 zUnionStoreKey2 WEIGHTS 2 3 AGGREGATE MIN
ZUNIONSTORE zUnionStoreMaxResult 2 zUnionStoreKey1 zUnionStoreKey2 WEIGHTS 2 3 AGGREGATE MAX
*
ZRANGE zUnionStoreSumResult 0 -1 WITHSCORES
ZRANGE zUnionStoreMinResult 0 -1 WITHSCORES
ZRANGE zUnionStoreMaxResult 0 -1 WITHSCORES
执行结果如下

下面是java代码
@Test
public void zUnionStore() {
zSetOperations.add("zUnionStoreKey1", "one", 1);
zSetOperations.add("zUnionStoreKey1", "two", 2);
zSetOperations.add("zUnionStoreKey2", "one", 1);
zSetOperations.add("zUnionStoreKey2", "two", 2);
zSetOperations.add("zUnionStoreKey2", "three", 3);
ZParams zParams = new ZParams();
zParams.weightsByDouble(2, 3);
zParams.aggregate(ZParams.Aggregate.SUM);
jedis.zunionstore("zUnionStoreSumResult", zParams, "zUnionStoreKey1", "zUnionStoreKey2");
//求最小值
zParams.aggregate(ZParams.Aggregate.MIN);
jedis.zunionstore("zUnionStoreMinResult", zParams, "zUnionStoreKey1", "zUnionStoreKey2");
//求最大值
zParams.aggregate(ZParams.Aggregate.MAX);
jedis.zunionstore("zUnionStoreMaxResult", zParams, "zUnionStoreKey1", "zUnionStoreKey2");
//spring
zSetOperations.unionAndStore("zUnionStoreKey1", "zUnionStoreKey2", "zUnionStoreResult");
printTuple("zUnionStoreSumResult", jedis.zrangeWithScores("zUnionStoreSumResult", 0, -1));
printTuple("zUnionStoreMinResult", jedis.zrangeWithScores("zUnionStoreMinResult", 0, -1));
printTuple("zUnionStoreMaxResult", jedis.zrangeWithScores("zUnionStoreMaxResult", 0, -1));
printTuple("zUnionStoreResult", jedis.zrangeWithScores("zUnionStoreResult", 0, -1));
}
还是那句话建议学习的人最好每个命令都去敲下,加深印象。
纸上得来终觉浅,绝知此事要躬行。————出自《冬夜读书示子聿》
redis系列:通过文章点赞排名案例学习sortedset命令的更多相关文章
- redis系列:通过共同好友案例学习set命令
前言 这一篇文章将讲述Redis中的set类型命令,同样也是通过demo来讲述,其他部分这里就不在赘述了. 项目Github地址:https://github.com/rainbowda/learnW ...
- redis系列:通过队列案例学习list命令
前言 这一篇文章将讲述Redis中的list类型命令,同样也是通过demo来讲述,其他部分这里就不在赘述了. 项目Github地址:https://github.com/rainbowda/learn ...
- redis系列:通过日志案例学习string命令
前言 该文章将通过一个小demo将讲述Redis中的string类型命令.demo将以springboot为后台框架快速开发,iview前端框架进行简单的页面设计,为了方便就不使用DB存储数据了,直接 ...
- redis系列:通过通讯录案例学习hash命令
前言 这一篇文章将讲述Redis中的hash类型命令,同样也是通过demo来讲述,其他部分这里就不在赘述了. 项目Github地址:https://github.com/rainbowda/learn ...
- Redis系列(十二):数据结构SortedSet跳跃表中基本操作命令和源码解析
1.SkipList Redis的sortedSet数据结构是有序不重复的(索引为唯一的,数据(score)却可以重复), 跳表是redis的一个核心组件,也同时被广泛地运用到了各种缓存地实现当中,它 ...
- redis 系列,这里转发别人博客, 和常用命令
https://blog.csdn.net/qq_35433716/category_7944890.html 常用命令: https://www.cnblogs.com/mznsndy/p/1395 ...
- Redis系列文章-数据结构篇
Redis系列文章 前言: 工作原因,在学习mybatis知识后,2个月没有补充新的知识了,最近拿起书本开始学习.打算写下这个Redis系列的文章. 目录结构如下: Redis内置数据结构 Redis ...
- redis系列:redis介绍与安装
前言 这个redis系列的文章将会记录博主学习redis的过程.基本上现在的互联网公司都会用到redis,所以学习这门技术于你于我都是有帮助的. 博主在写这个系列是用的是目前最新版本4.0.10,虚拟 ...
- Redis系列总结--这几点你会了吗?
文章原创于公众号:程序猿周先森.本平台不定时更新,喜欢我的文章,欢迎关注我的微信公众号. 前面几篇已经对Redis中几个关键知识点做了介绍,本篇主要对Redis系列做一下总结以及对Redis中常见面试 ...
随机推荐
- windows下Scrapy爬虫框架环境搭建
1. 安装python 根据你的需求下载python安装包,安装python(本文基于python27)https://www.python.org/downloads/ 下载完安装默认安装路径, ...
- POJ 2728 Desert King:最优比率生成树
题目链接:http://poj.org/problem?id=2728 题意: 给你n个点(x,y,z),让你求一棵生成树,使得 k = ∑ |z[i]-z[j]| / ∑ dis(i,j)最小. | ...
- POJ-1741(树分治)
树的点分治 给出详细的讲解!!点这里打开论文-分治算法在树的路径问题中的应用 本题目是他讲的第一个例题: 我的理解:每次都找树的重心,计算以重心为根的子树之间所贡献的答案.不断这样下去:如果这棵树是一 ...
- 关于MFC中重载函数是否调用基类相对应函数的问题
在重载CDialog的OnInitDialog()函数的时候,在首行会添加一句:CDialongEx::OnInitDialog();语句,这是为什么呢?什么时候添加,什么时候不添加? 实际上,我们在 ...
- codeforces 650 C. Watchmen(数学公式)
C. Watchmen time limit per test 3 seconds memory limit per test 256 megabytes input standard input o ...
- I.MX6 USB Camera
/************************************************************************* * I.MX6 USB Camera * 说明: ...
- [JSOI 2015] 最大公约数
[题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=4488 [算法] 不妨首先枚举左端点 注意到对于任意一个正整数n , 其质因子个数是l ...
- C# Hashtable赋值给另一个Hashtable时
c#中想将一个hashtable的元素全部给另一个hashbale时, 使用迭代一个一个元素赋值 如: ammus.Clear(); IDictionaryEnumerator ie = _temp. ...
- DevExpress源码编译总结
独家提供完整可编译sln文件,本篇文章内容包括基础知识(GAC.程序集强签名.友元程序集).编译过程.注册GAC.添加工具箱.多语言支持.运行时和设计时调试 源码地址 链接:http://pan.b ...
- mariadb的读写分离
实验环境:CentOS7 设备:一台主数据库服务器,两台从数据库服务器,一台调度器 主从的数据库配置请查阅:http://www.cnblogs.com/wzhuo/p/7171757.html : ...