当前位置:首页 > 科技  > 软件

Redis Zset详解:排行榜绝佳选择

来源: 责编: 时间:2024-05-16 09:02:48 249观看
导读最近我们发布了一款新的app,其中包含一个搜索功能。在搜索时,会给用户展示四个热门搜索词汇。我们利用 Redis 的有序集合(zset)实现了这一功能。由于应用程序刚刚上线并且尚未大力推广,所以热门搜索词汇显示的是我们随手测

最近我们发布了一款新的app,其中包含一个搜索功能。在搜索时,会给用户展示四个热门搜索词汇。我们利用 Redis 的有序集合(zset)实现了这一功能。由于应用程序刚刚上线并且尚未大力推广,所以热门搜索词汇显示的是我们随手测试词汇,如测试、test、111等。这会给人一种不够专业的印象。为了提升产品形象,我们计划通过后台删除这些测试的词汇,使热门搜索词汇更加贴近实际使用情况。今天,我将与大家分享在 Redis 命令行中操作有序集合(zset)的命令,以及我们实现热门搜索词汇功能的思路。ugm28资讯网——每日最新资讯28at.com

Redis ZSET 详解

Redis 中的 ZSET(有序集合)是一种有序的数据结构,它类似于 SET(集合),但每个成员都关联着一个分数(score),通过分数来进行排序。这使得 ZSET 既可以像 SET 一样快速查找成员,又可以按照分数从小到大或从大到小进行排序。ugm28资讯网——每日最新资讯28at.com

ZSET 的特点包括:ugm28资讯网——每日最新资讯28at.com

  • 有序性:成员按照分数的顺序排列,可以进行范围查询和排名操作。
  • 唯一性:每个成员都是唯一的,但不同成员可以有相同的分数。
  • 快速查找:和 SET 类似,ZSET 也可以在 O(1) 的时间复杂度内查找单个成员。
  • 分数(score)更新:可以对成员的分数进行增加或减少操作,同时保持排序。

ZSET 的底层实现会根据实际的情况选择ziplist(压缩列表)/listpack(紧凑列表)(redis7.0已经将 listpack 完整替代 ziplis) 或者skiplist(跳跃表),Redis 会根据实际情况动态地在这两种底层结构之间切换,使得其在内存和性能之间平衡。这是由两个配置参数:zset-max-ziplist-entries 和 zset-max-ziplist-value控制的,其默认值为128和64。当 Zset 存储的元素数量超过zset-max-ziplist-entries的值或者最长元素的长度超过 zset-max-ziplist-value的值的时候Redis 会将底层结构从压缩列表/紧凑列表转换为跳跃表。压缩列表/紧凑列表占用的内存比较少,但是修改数据时可能会对整个列表进行重写,性能较低; 跳跃表的查找和修改数据的性能较高,但是占用的内存也较多。ugm28资讯网——每日最新资讯28at.com

我们在redis 命令行中可以通过以下命令查看 zset的配置参数:ugm28资讯网——每日最新资讯28at.com

config get zset*

图片图片ugm28资讯网——每日最新资讯28at.com

Redis ZSET 使用场景

  • 排行榜

Redis 的zset是设计实时排行的绝佳选择,我们可以使用它来完成各种排行榜、热门词汇等场景的实现。我们app的热搜词汇也是通过zset实现的,本文中也将介绍热搜词汇的实现方式。ugm28资讯网——每日最新资讯28at.com

  • 延时队列

我们可以将时间戳设置为zset的score,延时处理的任务作为元素,定期或者循环扫描zset来处理到达时间的任务。ugm28资讯网——每日最新资讯28at.com

  • 滑动窗口限流

我们可以将接口地址设置为zset的key,时间戳设置为zset的score,使用uuid作为元素,那么我们可以通过zset获取到 score固定窗口范围的时间内的请求数来达到限流的目的。ugm28资讯网——每日最新资讯28at.com

REDISSON 操作ZSET数据

代码如下:ugm28资讯网——每日最新资讯28at.com

package cn.xj.xjdoc.redis.zset;import jakarta.annotation.Resource;import org.redisson.api.RScoredSortedSet;import org.redisson.api.RedissonClient;import org.redisson.client.protocol.ScoredEntry;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.stereotype.Service;import java.util.Collection;@Servicepublic class ZSETService {    private static final Logger log = LoggerFactory.getLogger(ZSETService.class);    @Resource    private RedissonClient redissonClient;    public void operation(){        String zsetKey = "xjzset";        RScoredSortedSet<String> zset = redissonClient.getScoredSortedSet(zsetKey);        //添加元素        zset.add(1.0, "修己xj1");        zset.add(2.0, "修己xj2");        zset.add(3.0, "修己xj3");        zset.add(4.0, "修己xj4");        // 获取ZSET中指定成员的分数        Double score = zset.getScore("修己xj2");        log.info("1、获取ZSET中指定成员的分数:{}",score);        //获取ZSET中指定成员的排名(分数从小到大排序)        Integer rank = zset.rank("修己xj3");        log.info("2、获取ZSET中指定成员的排名(分数从小到大排序):{}",rank);        //获取ZSET中指定成员的排名(分数从大到小排序)        Integer reverseRank = zset.revRank("修己xj4");        log.info("3、获取ZSET中指定成员的排名(分数从大到小排序):{}",reverseRank);        // 获取ZSET中指定排名范围内的成员(分数从小到大排序)        Collection<String> membersInRange = zset.valueRange(0, 1);        membersInRange.forEach(o->log.info("4、获取ZSET中指定排名范围内的成员(分数从小到大排序):{}",o));        // 获取ZSET中指定排名范围内的成员(分数从大到小排序)        Collection<String> membersInRangeRever = zset.valueRangeReversed(0, 1);        membersInRangeRever.forEach(o->log.info("5、获取ZSET中指定排名范围内的成员(分数从大到小排序):{}",o));        //获取ZSET中指定分数范围内的成员(分数从小到大排序)        Collection<String> membersInScoreRange = zset.valueRange(2.0, true, 3.0, true);        membersInScoreRange.forEach(o->log.info("6、获取ZSET中指定分数范围内的成员(分数从小到大排序):{}",o));        //获取ZSET中指定分数范围内的成员(分数从大到小排序)        Collection<String> membersInScoreRever = zset.valueRangeReversed(2.0, true, 3.0, true);        membersInScoreRever.forEach(o->log.info("7、获取ZSET中指定分数范围内的成员(分数从大到小排序):{}",o));        //获取ZSET中指定排名范围内的成员及其分数        Collection<ScoredEntry<String>> membersWithScoresInRange = zset.entryRange(0, 1);        membersWithScoresInRange.forEach(o->log.info("8、获取ZSET中指定排名范围内的成员及其分数,成员:{},分数",o.getValue(),o.getScore()));        //获取ZSET中指定分数范围内的成员及其分数        Collection<ScoredEntry<String>> membersWithScoresInScoreRange = zset.entryRange(3.0, true, 4.0, true);        membersWithScoresInScoreRange.forEach(o->log.info("9、获取ZSET中指定分数范围内的成员及其分数,成员:{},分数",o.getValue(),o.getScore()));        //        Double newScore = zset.addScore("修己xj4", 1);        log.info("10、增加1之后指定成员的分数:{}",newScore);        //删除ZSET 中的指定成员        Boolean removedFlag = zset.remove("修己xj3");        log.info("11、删除ZSET 中的指定成员:{}",removedFlag);        //删除指定排名范围内的成员        Integer removedByRangeCount = zset.removeRangeByRank(0, 1);        log.info("12、删除指定排名范围内的成员数量:{}",removedByRangeCount);        //删除指定分数范围内的成员        Integer removedByScoreCount = zset.removeRangeByScore(3.0, true, 4.0, true);        log.info("13、删除指定分数范围内的成员数量:{}",removedByScoreCount);    }}

执行结果如下:ugm28资讯网——每日最新资讯28at.com

1、获取ZSET中指定成员的分数:2.02、获取ZSET中指定成员的排名(分数从小到大排序):23、获取ZSET中指定成员的排名(分数从大到小排序):04、获取ZSET中指定排名范围内的成员(分数从小到大排序):修己xj14、获取ZSET中指定排名范围内的成员(分数从小到大排序):修己xj25、获取ZSET中指定排名范围内的成员(分数从大到小排序):修己xj45、获取ZSET中指定排名范围内的成员(分数从大到小排序):修己xj36、获取ZSET中指定分数范围内的成员(分数从小到大排序):修己xj26、获取ZSET中指定分数范围内的成员(分数从小到大排序):修己xj37、获取ZSET中指定分数范围内的成员(分数从大到小排序):修己xj37、获取ZSET中指定分数范围内的成员(分数从大到小排序):修己xj28、获取ZSET中指定排名范围内的成员及其分数,成员:修己xj1,分数8、获取ZSET中指定排名范围内的成员及其分数,成员:修己xj2,分数9、获取ZSET中指定分数范围内的成员及其分数,成员:修己xj3,分数9、获取ZSET中指定分数范围内的成员及其分数,成员:修己xj4,分数10、增加1之后指定成员的分数:5.011、删除ZSET 中的指定成员:true12、删除指定排名范围内的成员数量:213、删除指定分数范围内的成员数量:0

命令行操作ZSET数据

  • zadd 添加成员
zadd xjzset 1 "修己xj1" 2 "修己xj2" 3 "修己xj3" 4 "修己xj4"
  • zscore 获取指定成员的分数
zscore xjzset '修己xj2'
  • zrank 获取指定成员的排名(分数从小到大排序)
zrank xjzset 修己xj3
  • zrevrank 获取指定成员的排名(分数从大到小排序)
zrevrank xjzset 修己xj3
  • zrange/zrevrange 获取ZSET中指定排名范围内的成员   zrange:分数从小到大排序,我们加了一些测试数据,如下

图片图片ugm28资讯网——每日最新资讯28at.com

zrevrange:分数从大到小排序ugm28资讯网——每日最新资讯28at.com

图片图片ugm28资讯网——每日最新资讯28at.com

zrange key start stop [withscores]zrevrange key start stop [withscores]

其中,key是zset的键名,start是起始索引,stop结束索引,withscores表示是否同时返回分数。可以使用负数索引表示从末尾开始,比如-1表示最后一个元素。zrange key 0 -1 则会显示出所有元素ugm28资讯网——每日最新资讯28at.com

zrange xjzset 1 2 withscores
  • zrangebyscore/zrevrangebyscore 获取ZSET中指定分数score范围内的成员  zrangebyscore:分数从小到大排序,zrevrangebyscore:分数从大到小排序
zrangebyscore key min max [withscores]zrevrangebyscore key max min [withscores]

其中,key是zset的键名,min 和 max 表示score的范围,范围为闭区间,withscores表示是否同时返回分数。ugm28资讯网——每日最新资讯28at.com

zrangebyscore xjzset 2.5 3.5 withscores
  • zincrby 将指定成员的分数增加指定的值
zincrby xjzset  1  修己xj3

注: 进行double的值的运算时可能会丢失精度,如果对score进行运算时尽可能使用整数运算。ugm28资讯网——每日最新资讯28at.com

图片图片ugm28资讯网——每日最新资讯28at.com

  • zcard 返回zset中成员的数量
zcard xjzset
  • zcount 获取指定范围分数内的成员的数量
zcount key min max

其中,key是zset的键名,min 和 max 表示score的范围,范围为闭区间。ugm28资讯网——每日最新资讯28at.com

zcount  xjzset 0 2
  • zrem 删除指定成员
zrem  xjzset
  • zremrangebyrank  删除指定排名范围内的成员
zremrangebyrank key start stop

其中,key是zset的键名,start是起始索引,stop结束索引。ugm28资讯网——每日最新资讯28at.com

zremrangebyrank xjzset 1 1
  • zremrangebyscore 删除指定分数范围内的成员
zremrangebyscore key min max

其中,key是zset的键名,min 和 max 表示score的范围,范围为闭区间。ugm28资讯网——每日最新资讯28at.com

zremrangebyscore  xjzset 0 3

图片图片ugm28资讯网——每日最新资讯28at.com

热搜词汇功能实现

我们设计思路是 将每个搜索词作为有序集合的成员,而搜索次数作为成员的分数,每次搜索的时候对这个搜索词的分数加1,这样可以根据搜索次数对热搜词进行排序。ugm28资讯网——每日最新资讯28at.com

  • 搜索接口
public String keySearch(String keyStr){    String hotSearchKey = "xj_hotSearch";    RScoredSortedSet<String> hotSearchZSet = redissonClient.getScoredSortedSet(hotSearchKey);    //更新zset中当前搜索词的搜索次数    hotSearchZSet.addScore(keyStr,1);    //搜索逻辑    //doSearch(keyStr);    return keyStr;}
  • 热搜词汇查询接口
public Collection<String> hotSearch(){    String hotSearchKey = "xj_hotSearch";    RScoredSortedSet<String> hotSearchZSet = redissonClient.getScoredSortedSet(hotSearchKey);    //获取zset中点击次数排名前5的数据    Collection<String> hotList= hotSearchZSet.valueRangeReversed(0,4);    return hotList;}

我们加了一些测试数据,如下ugm28资讯网——每日最新资讯28at.com

图片图片ugm28资讯网——每日最新资讯28at.com

图片图片ugm28资讯网——每日最新资讯28at.com

总结

通过本文的介绍,你学会了如何利用Spring Boot和Redis的ZSET数据结构实现热门搜索功能,并深入了解了热搜词汇的实现细节。通过合理的设计和优化,可以为用户提供更好的搜索体验,同时也提升了应用程序的性能和可扩展性。ugm28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-88319-0.htmlRedis Zset详解:排行榜绝佳选择

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 一文彻底搞明白备忘录模式

下一篇: 盘点GoLang中的三方库:govaluate、flag、go-homedir、cast

标签:
  • 热门焦点
  • 6月iOS设备好评榜:第一蝉联榜首近一年

    作为安兔兔各种榜单里变化最小的那个,2023年6月的iOS好评榜和上个月相比没有任何排名上的变化,仅仅是部分设备好评率的下降,长年累月的用户评价和逐渐退出市场的老款机器让这
  • 虚拟键盘 API 的妙用

    你是否在遇到过这样的问题:移动设备上有一个固定元素,当激活虚拟键盘时,该元素被隐藏在了键盘下方?多年来,这一直是 Web 上的默认行为,在本文中,我们将探讨这个问题、为什么会发生
  • 小红书1周涨粉49W+,我总结了小白可以用的N条涨粉笔记

    作者:黄河懂运营一条性教育视频,被54万人&ldquo;珍藏&rdquo;是什么体验?最近,情感博主@公主是用鲜花做的,火了!仅仅凭借一条视频,光小红书就有超过128万人,为她疯狂点赞!更疯狂的是,这
  • Temu起诉SHEIN,跨境电商战事升级

    来源 | 伯虎财经(bohuFN)作者 | 陈平安日前据外媒报道,拼多多旗下跨境电商平台Temu正对竞争对手SHEIN提起新诉讼,诉状称Shein&ldquo;利用市场支配力量强迫服装厂商与之签订独家
  • 共享单车的故事讲到哪了?

    来源丨海克财经与共享充电宝相差不多,共享单车已很久没有被国内热点新闻关照到了。除了一再涨价和用户直呼用不起了。近日多家媒体再发报道称,成都、天津、郑州等地多个共享单
  • 大厂卷向扁平化

    来源:新熵作者丨南枝 编辑丨月见大厂职级不香了。俗话说,兵无常势,水无常形,互联网企业调整职级体系并不稀奇。7月13日,淘宝天猫集团启动了近年来最大的人力制度改革,目前已形成一
  • 3699元!iQOO Neo8 Pro顶配版今日首销:1TB UFS 4.0同价位唯一

    5月23日,iQOO推出了全新的iQOO Neo8系列,包含iQOO Neo8和iQOO Neo8 Pro两个版本,其中标准版搭载高通骁龙8+,而Pro版更是首发搭载了联发科天玑9200+旗舰
  • 联想小新Pad Pro 12.6将要推出,搭载高通骁龙 870 处理器

    联想小新Pad Pro 12.6将于秋季新品会上推出,官方按照惯例直接在发布会前给出了机型的所有参数。联想小新 Pad Pro 12.6 将搭载高通骁龙 870 处理器,重量为 5
  • 微软发布Windows 11新版 引入全新任务栏状态

    近日,微软发布了Windows 11新版,而Build 22563更新主要引入了几周前曝光的平板模式任务栏等,系统更流畅了。更新中,Windows 11加入了专门针对平板优化的任务栏
Top