• 基于SpringBoot+Redis实现点赞/排行榜功能,可同理实现收藏/关注功能,可拓展实现共同好友/共同关注/关注推送功能


    前言

    简单记录一下在SpringBoot项目中,使用Redis实现点赞/排行榜功能,可同理实现收藏/关注功能,可拓展实现共同好友/共同关注/关注推送功。主要用到了Redis中的Set集合和ZSet集合。

    一、指定使用某个索引的数据库

    在Redis中,可以使用SELECT命令来选择要使用的数据库索引。默认情况下,Redis有16个数据库索引,编号从0到15。

    SELECT 15

    二、Redis集合(Set)数据类型的SADD、SREM、SMEMBERS命令

    1.SADD 命令

    (1)用法:SADD key val_1 或 SADD key val_1, val_2 ...
    (2)作用:将一个或多个成员元素加入到集合中,已经存在于集合的成员元素将被忽略。
    (3)返回值:被添加到集合中的新元素的数量,不包括被忽略的元素。
    (4)示例

    1. redis > SADD myset "hello"
    2. (integer) 1
    3. redis > SADD myset "foo"
    4. (integer) 1
    5. redis > SADD myset "hello"
    6. (integer) 0
    7. redis > SMEMBERS myset
    8. 1) "hello"
    9. 2) "foo"

    2.SREM 命令

    (1)用法:SREM key val_1 或 SREM key val_1, val_2 ...
    (2)作用:移除集合中一个或多个成员
    (3)返回值:被成功移除的元素的数量,不包括被忽略的元素。
    (4)示例

    1. redis > SADD myset1 "hello"
    2. (integer) 1
    3. redis > SADD myset1 "world"
    4. (integer) 1
    5. redis > SADD myset1 "bar"
    6. (integer) 1
    7. redis > SREM myset1 "hello"
    8. (integer) 1
    9. redis > SREM myset1 "foo"
    10. (integer) 0
    11. redis > SMEMBERS myset1
    12. 1) "bar"
    13. 2) "world"

    3.SMEMBERS 命令

    (1)用法:SMEMBERS key
    (2)作用:返回集合中的所有的成员。
    (3)返回值:集合中的所有成员。

    4.SISMEMBER 命令

    (1)用法:SISMEMBER key member
    (2)作用:判断成员元素是否是集合的成员。
    (3)返回值:如果成员元素是集合的成员,返回 1 。 如果成员元素不是集合的成员,或 key 不存在,返回 0 。
    (4)示例

    1. redis > SADD myset1 "hello"
    2. (integer) 1
    3. redis > SISMEMBER myset1 "hello"
    4. (integer) 1
    5. redis > SISMEMBER myset1 "world"
    6. (integer) 0

    5.SINTERSTORE 命令

    (1)用法:SINTERSTORE DESTINATION_KEY KEY1 KEY2 ...
    (2)作用:将给定集合之间的交集存储在指定的集合中。
    (3)返回值:返回存储交集的集合的元素数量。
    (4)示例

    1. redis > SADD myset1 "hello"
    2. (integer) 1
    3. redis > SADD myset1 "foo"
    4. (integer) 1
    5. redis > SADD myset1 "bar"
    6. (integer) 1
    7. redis > SADD myset2 "hello"
    8. (integer) 1
    9. redis > SADD myset2 "world"
    10. (integer) 1
    11. redis > SINTERSTORE myset myset1 myset2
    12. (integer) 1
    13. redis > SMEMBERS myset
    14. 1) "hello"

    (5)场景:实现共同好友、共同关注、关注推送等功能。

    三、Redis集合(ZSet)数据类型的ZINCRBY、ZRANGE、ZREVRANGE、ZSCORE命令

    1.ZINCRBY 命令

    (1)用法:ZINCRBY key increment member
    (2)作用:对有序集合中指定成员的分数加上增量increment,
    · 可以通过传递一个负数值increment,让分数减去相应的值;
    · 当key不存在,或分数不是key的成员时,ZINCRBY key increment member 等同于 ZADD key increment member;
    · 当key不是有序集类型时,返回一个错误;
    (3)返回值:member成员的新分数值。
    (4)示例

    1. redis > ZADD myzset 1 "one"
    2. (integer) 1
    3. redis > ZADD myzset 2 "two"
    4. (integer) 1
    5. redis > ZINCRBY myzset 2 "one"
    6. "3"
    7. redis > ZRANGE myzset 0 -1 WITHSCORES
    8. 1) "two"
    9. 2) "2"
    10. 3) "one"
    11. 4) "3"

    2.ZRANGE 命令

    (1)用法:ZRANGE key start stop [WITHSCORES]
    (2)作用:返回有序集中,指定区间内的成员,其中成员的位置按分数值递增(从小到大)来排序。
    (3)返回值:指定区间内,带有分数值(可选)的有序集成员的列表。
    (4)示例

    1. redis > ZRANGE Blog-Rank 0 -1 WITHSCORES
    2. 1) "Blog-10"
    3. 2) "1"
    4. 3) "Blog-1"
    5. 4) "2"
    6. 5) "Blog-5"
    7. 6) "3"

    3.ZREVRANGE 命令

    (1)用法:ZREVRANGE key start stop [WITHSCORES]
    (2)作用:返回有序集中,指定区间内的成员,其中成员的位置按分数值递减(从大到小)来排列。
    (3)返回值:指定区间内,带有分数值(可选)的有序集成员的列表。
    (4)示例

    1. redis > ZREVRANGE Blog-Rank 0 -1 WITHSCORES
    2. 1) "Blog-5"
    3. 2) "3"
    4. 3) "Blog-1"
    5. 4) "2"
    6. 5) "Blog-10"
    7. 6) "1"

    4.ZSCORE 命令

    (1)用法:ZSCORE key member
    (2)作用:返回有序集中,成员的分数值。 如果成员元素不是有序集 key 的成员,或 key 不存在,返回 nil 。
    (3)返回值:成员的分数值,以字符串形式表示。
    (4)示例

    1. redis > ZRANGE salary 0 -1 WITHSCORES
    2. 1) "tom"
    3. 2) "2000"
    4. 3) "peter"
    5. 4) "3500"
    6. 5) "jack"
    7. 6) "5000"
    8. redis > ZSCORE salary peter
    9. "3500"

    四、示例代码

    1.控制层

    (1)UserController.java

    1. /**
    2. * 点赞
    3. * 同一个用户只能点赞一次,再次点击则取消点赞,若当前用户已经点赞,则点赞按钮高亮显示
    4. */
    5. @PutMapping(value = "like/{blogId}")
    6. @ResponseBody
    7. @CrossOrigin
    8. public T like (@PathVariable("blogId") Long blogId) {
    9. return userService.like(blogId);
    10. }
    11. /**
    12. * 排行榜
    13. * 查询点赞量最多的3篇博文
    14. */
    15. @GetMapping(value = "blogTop")
    16. @ResponseBody
    17. @CrossOrigin
    18. public T blogTop () {
    19. return userService.blogTop();
    20. }

    2.接口层

    (1)IUserService.java

    1. T like(Long blogId);
    2. T blogTop();

    3.实现层

    (1)UserServiceImpl.java

    1. private static final String BLOG_LIKED_KEY = "Blog-Liked-";
    2. private static final String BLOG_RANK_KEY = "Blog-Rank";
    3. private static final String BLOG_KEY = "Blog-";
    4. @Autowired
    5. private StringRedisTemplate stringRedisTemplate;
    6. @Override
    7. public T like(Long blogId) {
    8. HashMap responseObj = new HashMap<>();
    9. // 获取登录用户
    10. UserDTO userDTO = RequestHolder.getUser();
    11. // 是否已点赞
    12. String key = BLOG_LIKED_KEY + blogId; // Blog-Liked-10
    13. String val = userDTO.getPhone(); // 13800138000
    14. Boolean isMember = stringRedisTemplate.opsForSet().isMember(key, val); // SISMEMBER Blog-Liked-10 "13800138000"
    15. if (BooleanUtil.isFalse(isMember)) {
    16. // 未点赞
    17. boolean isSuccess = true; // 在数据库点赞表中,新增/修改关于此博文的点赞状态为1
    18. if (isSuccess) {
    19. stringRedisTemplate.opsForSet().add(key, val); // SADD Blog-Liked-10 "13800138000"
    20. stringRedisTemplate.opsForZSet().incrementScore(BLOG_RANK_KEY, BLOG_KEY + blogId, 1); // ZINCRBY BLOG_RANK_KEY 1 Blog-10
    21. }
    22. } else {
    23. // 已点赞
    24. boolean isSuccess = true; // 在数据库点赞表中,修改关于此博文的点赞状态为0
    25. if (isSuccess) {
    26. stringRedisTemplate.opsForSet().remove(key, val); // SREM Blog-Liked-10 "13800138000"
    27. stringRedisTemplate.opsForZSet().incrementScore(BLOG_RANK_KEY, BLOG_KEY + blogId, -1); // ZINCRBY BLOG_RANK_KEY -1 Blog-10
    28. }
    29. }
    30. responseObj.put("code", 200);
    31. responseObj.put("success", true);
    32. return (T) responseObj;
    33. }
    34. @Override
    35. public T blogTop() {
    36. HashMap responseObj = new HashMap<>();
    37. // Set> set = stringRedisTemplate.opsForZSet().rangeWithScores(BLOG_RANK_KEY, 0, -1); // ZRANGE Blog-Rank 0 -1 WITHSCORES
    38. Set> set = stringRedisTemplate.opsForZSet().reverseRangeWithScores(BLOG_RANK_KEY, 0, 2); // ZREVRANGE Blog-Rank 0 2 WITHSCORES
    39. System.out.println("blogTop :: set -> " + set);
    40. List list = new ArrayList<>();
    41. for (ZSetOperations.TypedTuple tuple : set) {
    42. HashMap map = new HashMap();
    43. String key = tuple.getValue();
    44. double score = tuple.getScore();
    45. long val = (long) score;
    46. map.put(key, val);
    47. list.add(map);
    48. }
    49. responseObj.put("code", 200);
    50. responseObj.put("success", true);
    51. responseObj.put("data", list);
    52. return (T) responseObj;
    53. }

    五、其它相关知识点

    1.stringRedisTemplate.keys(pattern)方法说明

    (1)参数说明
    - 匹配所有键:*
    - 匹配以特定前缀开头的键:prefix*
    - 匹配以特定后缀结尾的键:*suffix
    - 匹配包含特定字符串的键:*substring*
    - 匹配特定模式的键:pattern?
    (2)示例

    1. // 查询所有key列表
    2. Set keys = stringRedisTemplate.keys("*");
    3. System.out.println("blogTop :: keys -> " + keys);
    4. // 查询点赞博文的key列表
    5. Set blogKeys = stringRedisTemplate.keys(BLOG_LIKED_KEY + "*");
    6. System.out.println("blogTop :: blogKeys -> " + blogKeys);

  • 相关阅读:
    mp4转换成gif怎么转?
    java.awt.HeadlessException
    APS在印刷行业的应用前景和应用效益
    浏览器中环境判断和系统判断-js封装
    秋招—文思海辉笔试题
    简单几步便可轻松制作思维导图,快来查收这份干货
    卫星通信系统按照工作轨道分类
    微信小程序使用微信云托管添加自定义域名并转发到pexels.com
    java计算机毕业设计学生勤工助学管理系统源程序+mysql+系统+lw文档+远程调试
    国自然中标越来越难,怎样才能赢在起跑线上?
  • 原文地址:https://blog.csdn.net/Cai181191/article/details/134552302