用Redis做数据排名

1.背景

用Redis做数据缓存用的比较多,大家都能熟练使用String和Hash结构去存储数据,今天讲下如何使用ZSet来做数据排名。

假设场景是需要按天存储全国城市的得分数据,可以查询前十名的城市排名。

这个case可以使用传统关系型数据库做,设计一个表存储城市名、得分、录入时间,查询的时候按时间过滤某天的数据然后排序取前10条,这样做的缺点是存储的数据会越来越多,后面的查询会越来越慢。选择ZSet可以自动排序,并且key设置过期时间后,可以仅仅保存最近多少天的数据。

2.数据结构设计

  • key:选择日期为key,比如20230924
  • member:用城市名称作为value,比如changsha
  • score:计算出来的值

3.接口设计

3.1 新增数据接口

该接口负责将城市得分放到每日的key中。

java 复制代码
@GetMapping("/redis/set")
    public void set(@RequestParam("key") String key, @RequestParam("city") String city,@RequestParam("value") String value){
        Long total = redisTemplate.opsForZSet().zCard(key);

        if (total >= 10) {
            return;
        }

        redisTemplate.opsForZSet().add(key, city, Double.parseDouble(value));
        if (redisTemplate.hasKey(key) && 0 == total) {
            redisTemplate.expire(key, 7, TimeUnit.DAYS);
        }
    }
3.2 查询城市得分排名接口
java 复制代码
@GetMapping("/redis/get")
    public String get(@RequestParam("key") String key){
        Set<ZSetOperations.TypedTuple<Object>> typedTuples = redisTemplate.opsForZSet().reverseRangeWithScores(key, 0, 9);
        List<CityRank> list = new ArrayList<>();
        typedTuples.stream().forEach(
                tuple -> {
                    CityRank cityRank = new CityRank();
                    cityRank.setScore(tuple.getScore());
                    cityRank.setValue(tuple.getValue().toString());
                    list.add(cityRank);
                });
        return JSON.toJSONString(list);
    }

4.测试

使用第一个接口录入数据,得到的结果:

使用第二个接口查询城市排名:

java 复制代码
[{"score":770.0,"value":"nanchang"},{"score":750.0,"value":"guangzhou"},{"score":550.0,"value":"xiamen"},{"score":500.0,"value":"nanjing"}]

5.总结

在使用ZSet的时候设置Key过期时间想了好久,如果在Key没有创建的时候对其设置过期时间是无效的,因此在添加完数据后做了一个判断,必须有Key,并且里面只有一个元素这是就设置过期时间,后面增加数据不会再走此逻辑,防止延长过期时间导致Key一直存在。希望朋友们看了我的文章针对数据排名展示这块有更好的想法和我交流。

相关推荐
qq_372154233 分钟前
CSS如何改变单个网格项目的对齐方式
jvm·数据库·python
kexnjdcncnxjs3 分钟前
CodeIgniter4安全加固指南:防御XSS与CSRF攻击
jvm·数据库·python
2401_871492854 分钟前
Imagick PDF 处理失败的常见原因与解决方案
jvm·数据库·python
小年糕是糕手7 分钟前
【C/C++刷题集】栈、stack、队列、queue核心精讲
c语言·开发语言·数据结构·数据库·c++·算法·蓝桥杯
鸽芷咕12 分钟前
KingbaseES与Oracle兼容性深度解析:数据类型、内置函数、PL/SQL全面对比
数据库·sql·oracle
2301_8084143821 分钟前
MySQL数据类型
数据库·mysql
minji...21 分钟前
Linux 线程同步与互斥(六) 线程安全与重入问题,死锁,线程done
linux·运维·开发语言·数据库·c++·算法·安全
小羽网安2 小时前
从零开始学习 sql 注入,常见的 sql 注入解析
数据库·sql·学习
2401_846339563 小时前
CSS如何优化大型项目样式_使用SASS预处理器提升开发效率
jvm·数据库·python
ss27310 小时前
食谱推荐系统功能测试如何写?
java·数据库·spring boot·功能测试