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

基于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)示例

redis > SADD myset "hello"
(integer) 1
redis > SADD myset "foo"
(integer) 1
redis > SADD myset "hello"
(integer) 0
redis > SMEMBERS myset
1) "hello"
2) "foo"

2.SREM 命令

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

redis > SADD myset1 "hello"
(integer) 1
redis > SADD myset1 "world"
(integer) 1
redis > SADD myset1 "bar"
(integer) 1
redis > SREM myset1 "hello"
(integer) 1
redis > SREM myset1 "foo"
(integer) 0
redis > SMEMBERS myset1
1) "bar"
2) "world"

3.SMEMBERS 命令

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

4.SISMEMBER 命令

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

redis > SADD myset1 "hello"
(integer) 1
redis > SISMEMBER myset1 "hello"
(integer) 1
redis > SISMEMBER myset1 "world"
(integer) 0

5.SINTERSTORE 命令

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

redis > SADD myset1 "hello"
(integer) 1
redis > SADD myset1 "foo"
(integer) 1
redis > SADD myset1 "bar"
(integer) 1
redis > SADD myset2 "hello"
(integer) 1
redis > SADD myset2 "world"
(integer) 1
redis > SINTERSTORE myset myset1 myset2
(integer) 1
redis > SMEMBERS myset
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)示例

redis > ZADD myzset 1 "one"
(integer) 1
redis > ZADD myzset 2 "two"
(integer) 1
redis > ZINCRBY myzset 2 "one"
"3"
redis > ZRANGE myzset 0 -1 WITHSCORES
1) "two"
2) "2"
3) "one"
4) "3"

2.ZRANGE 命令

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

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

3.ZREVRANGE 命令

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

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

4.ZSCORE 命令

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

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

四、示例代码

1.控制层

(1)UserController.java

/*** 点赞* 同一个用户只能点赞一次,再次点击则取消点赞,若当前用户已经点赞,则点赞按钮高亮显示*/
@PutMapping(value = "like/{blogId}")
@ResponseBody
@CrossOrigin
public <T> T like (@PathVariable("blogId") Long blogId) {return userService.like(blogId);
}/*** 排行榜* 查询点赞量最多的3篇博文*/
@GetMapping(value = "blogTop")
@ResponseBody
@CrossOrigin
public <T> T blogTop () {return userService.blogTop();
}

2.接口层

(1)IUserService.java

<T> T like(Long blogId);<T> T blogTop();

3.实现层

(1)UserServiceImpl.java

private static final String BLOG_LIKED_KEY = "Blog-Liked-";
private static final String BLOG_RANK_KEY = "Blog-Rank";
private static final String BLOG_KEY = "Blog-";@Autowired
private StringRedisTemplate stringRedisTemplate;@Override
public <T> T like(Long blogId) {HashMap<String, Object> responseObj = new HashMap<>();// 获取登录用户UserDTO userDTO = RequestHolder.getUser();// 是否已点赞String key = BLOG_LIKED_KEY + blogId; // Blog-Liked-10String val = userDTO.getPhone(); // 13800138000Boolean isMember = stringRedisTemplate.opsForSet().isMember(key, val); // SISMEMBER Blog-Liked-10 "13800138000"if (BooleanUtil.isFalse(isMember)) {// 未点赞boolean isSuccess = true; // 在数据库点赞表中,新增/修改关于此博文的点赞状态为1if (isSuccess) {stringRedisTemplate.opsForSet().add(key, val); // SADD Blog-Liked-10 "13800138000"stringRedisTemplate.opsForZSet().incrementScore(BLOG_RANK_KEY, BLOG_KEY + blogId, 1); // ZINCRBY BLOG_RANK_KEY 1 Blog-10}} else {// 已点赞boolean isSuccess = true; // 在数据库点赞表中,修改关于此博文的点赞状态为0if (isSuccess) {stringRedisTemplate.opsForSet().remove(key, val); // SREM Blog-Liked-10 "13800138000"stringRedisTemplate.opsForZSet().incrementScore(BLOG_RANK_KEY, BLOG_KEY + blogId, -1); // ZINCRBY BLOG_RANK_KEY -1 Blog-10}}responseObj.put("code", 200);responseObj.put("success", true);return (T) responseObj;
}@Override
public <T> T blogTop() {HashMap<String, Object> responseObj = new HashMap<>();// Set<ZSetOperations.TypedTuple<String>> set = stringRedisTemplate.opsForZSet().rangeWithScores(BLOG_RANK_KEY, 0, -1); // ZRANGE Blog-Rank 0 -1 WITHSCORESSet<ZSetOperations.TypedTuple<String>> set = stringRedisTemplate.opsForZSet().reverseRangeWithScores(BLOG_RANK_KEY, 0, 2); // ZREVRANGE Blog-Rank 0 2 WITHSCORESSystem.out.println("blogTop :: set -> " + set);List<HashMap> list = new ArrayList<>();for (ZSetOperations.TypedTuple<String> tuple : set) {HashMap<String, Long> map = new HashMap();String key = tuple.getValue();double score = tuple.getScore();long val = (long) score;map.put(key, val);list.add(map);}responseObj.put("code", 200);responseObj.put("success", true);responseObj.put("data", list);return (T) responseObj;
}

五、其它相关知识点

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

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

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

相关文章:

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

前言 简单记录一下在SpringBoot项目中&#xff0c;使用Redis实现点赞/排行榜功能&#xff0c;可同理实现收藏/关注功能&#xff0c;可拓展实现共同好友/共同关注/关注推送功。主要用到了Redis中的Set集合和ZSet集合。 一、指定使用某个索引的数据库 在Redis中&#xff0c;可…...

AI“胡说八道”?怎么解?

原创 | 文 BFT机器人 01 引言 近年来&#xff0c;人工智能产业迅猛发展&#xff0c;大型语言模型GPT-4发展势头强劲&#xff0c;OpenAI推出ChatGPT、微软推出Bing、马斯克推出“最好的聊天机器人Grok”……科技巨头纷纷入局AI领域&#xff0c;引入人工智能作为办公工具的行业…...

[SIGGRAPH-23] 3D Gaussian Splatting for Real-Time Radiance Field Rendering

pdf | proj | code 本文提出一种新的3D数据表达形式3D Gaussians。每个Gaussian由以下参数组成&#xff1a;中心点位置、协方差矩阵、可见性、颜色。通过世界坐标系到相机坐标系&#xff0c;再到图像坐标系的仿射关系&#xff0c;可将3D Gaussian映射到相机坐标系&#xff0c;通…...

大话设计模式C++实现

大话设计模式&#xff0c;讲得非常好&#xff0c;但是作者是用C#写的&#xff0c;为了方便C程序员&#xff0c;使用C写了大话设计模式的代码 详情见Github&#xff1a;https://github.com/liubamboo/BigTalkDesignPattern...

IT 领域中的主要自动化趋势

48%的IT自动化流程属于IT服务管理&#xff0c;过去一年中&#xff0c;IT运维自动化增长了272%。 IT部门从交付者转变为战略伙伴 今年的《工作自动化指数》数据显示&#xff0c;自动化正在蔓延到组织的各个部门&#xff0c;越来越多的部门采用自动化&#xff0c;并且IT以外的员工…...

使用Python解析CAN总线

缘起 在新能源车辆的开发和维护中&#xff0c;经常需要对CAN总线数据进行分析。CANOE等总线软件虽然方便&#xff0c;但功能有限&#xff0c;难以满足数据分析的要求。Matlab的Vehicle Network Toolbox可以方便的进行数据解析和分析&#xff0c;它是闭源且收费的。因此&#x…...

DevExpress中文教程 - 如何在macOS和Linux (CTP)上创建、修改报表(下)

DevExpress Reporting是.NET Framework下功能完善的报表平台&#xff0c;它附带了易于使用的Visual Studio报表设计器和丰富的报表控件集&#xff0c;包括数据透视表、图表&#xff0c;因此您可以构建无与伦比、信息清晰的报表。 DevExpress Reports — 跨平台报表组件&#x…...

RAID的应用场景以及优缺点

RAID 0(条带化)&#xff1a; 工作原理&#xff1a; 数据被分成块&#xff0c;每个块写入不同的驱动器&#xff0c;以并行方式提高读写性能。 优势&#xff1a; 卓越的性能提升&#xff0c;特别是对于大型文件的读写操作。 劣势&#xff1a; 完全没有冗余&#xff0c;一个驱动器…...

java SpringCloud版本b2b2c鸿鹄云商平台全套解决方案 小程序商城免费搭建

使用技术&#xff1a; Spring CloudSpring BootMybatis微服务服务监控可视化运营 B2B2C平台&#xff1a; 平台管理端(包含自营) 商家平台端(多商户入驻) PC买家端、手机wap/公众号买家端 微服务&#xff08;30个通用微服务如&#xff1a;商品、订单、购物车、个人中心、支…...

[Linux] shell脚本的函数和数组

一、函数 1.1 函数的定义 函数是脚本的别名 作用&#xff1a;函数可以避免代码重复&#xff0c;可读性强&#xff0c;可以简化脚本。 格式&#xff1a;函数名&#xff08;&#xff09;{脚本} 1.2 如何使用函数 1.定义 2.调用 函数一定要先定义再使用 例子&#xff1a…...

万宾科技智能井盖的效果怎么样?

日常出行过程中&#xff0c;人们最不想看到交通拥堵或者道路维修等现象&#xff0c;因为这代表出行受到影响甚至会导致不能按时赴约等。所以城市路面的安全和稳定&#xff0c;是市民朋友非常关心的话题。骑行在路上的时候&#xff0c;如果经过井盖时发出异常声响&#xff0c;骑…...

nvm切换版本之后npm用不了

原因是 nvm只给你安了对应的node没给你安装对应的node版本的npm 解决办法如下 1找到你安装的node版本号 然后去官网下载对应的版本包 这个网址就是node官网的版本列表 Index of /download/release/ 2下载后解压 把根目录这俩复制到自己的nvm安装目录下 还有那个node_modul…...

【elementui】el-popover在列表里循环使用,取消的doClose无效解决办法

目录 一、需求效果二、代码详情html方法接口 一、需求效果 在使用elementui的Popover 弹出框时&#xff0c;需求是在table列表里使用&#xff0c;循环出来&#xff0c;无法取消。 二、代码详情 html <el-table-column v-if"checkPermission([admin,user:resetPass…...

【php】解决加载速度慢导出超时问题记录

...

postgresql安装fdw扩展

最近有同一个服务器不同数据库、不同服务器数据库之间的数据同步需求&#xff0c;使用了fdw 下面举例的是同一个服务器两个不同数据库的同步情况 1、安装扩展 create extension postgres_fdw; 在需要使用fdw的数据库都加上该扩展 2、创建fdw服务器 mlhbase_prd库 CREATE…...

反爬虫机制与反爬虫技术(二)

反爬虫机制与反爬虫技术二 1、动态页面处理与验证码识别概述2、反爬虫案例:页面登录与滑块验证码处理2.1、用例简介2.2、库(模块)简介2.3、网页分析2.4、Selenium准备操作2.5、页面登录2.6、模糊移动滑块测试3、滑块验证码处理:精确移动滑块3.1、精确移动滑块的原理3.2、滑…...

Grails 启动

Grails系列 Grails项目启动 文章目录 Grails系列Grails一、项目创建二、可能的问题1.依赖下载2.项目导入到idea失败3.项目导入到idea后运行报错 Grails Grails是一款基于Groovy语言的Web应用程序框架&#xff0c;它使用了许多流行的开源技术&#xff0c;如Spring Framework、…...

2023年亚太地区数学建模大赛 问题A

采果机器人的图像识别技术 中国是世界上最大的苹果生产国&#xff0c;年产量约为3500万吨。与此同时&#xff0c;中国也是世界上最大的苹果出口国&#xff0c;全球每两个苹果中就有一个&#xff0c;全球超过六分之一的苹果出口自中国。中国提出了一带一路倡议&#xff08;BRI&…...

基于springboot实现校园在线拍卖系统项目【项目源码】

基于springboot实现校园在线拍卖系统演示 Javar技术 JavaScript是一种网络脚本语言&#xff0c;广泛运用于web应用开发&#xff0c;可以用来添加网页的格式动态效果&#xff0c;该语言不用进行预编译就直接运行&#xff0c;可以直接嵌入HTML语言中&#xff0c;写成js语言&…...

详解ES6的Promise

ES6&#xff08;ECMAScript 6&#xff09;是JavaScript的一种标准&#xff0c;也被称为ES2015。它是在2015年发布的第六个ECMAScript标准版本&#xff0c;引入了许多新的语法和特性来增强JavaScript的功能和可读性。 文章目录 一、创建promise 二、处理Promise 三.Promise链…...

终极指南:Agent创业机会,从垂直领域切入

终极指南&#xff1a;Agent创业机会&#xff0c;从垂直领域切入 引言 痛点引入&#xff1a;AI Agent浪潮下&#xff0c;普通创业者/技术人该怎么抢滩&#xff1f; 过去两年的AI技术发展&#xff0c;完全是“天翻地覆”级别的——ChatGPT的诞生把通用大语言模型&#xff08;LLM&…...

第33篇:超参数调优实战——用网格搜索与随机搜索为模型“精调”(项目实战)

文章目录项目背景技术选型架构设计核心实现第一步&#xff1a;封装训练评估过程第二步&#xff1a;定义搜索空间并执行搜索踩坑记录效果对比项目背景 在之前的实战中&#xff0c;我们搭建了一个基础的图像分类模型&#xff0c;用的是经典的ResNet架构和CIFAR-10数据集。模型跑…...

CT1832 Real.Pi开发板:边缘AI与计算机视觉实战指南

1. CT1832 Real.Pi开发板深度解析Centron Design推出的CT1832 Real.Pi开发板&#xff0c;采用Realtek RTD1619B SoC&#xff0c;完美兼容树莓派3 Model B的外形尺寸。这块板子最吸引我的地方在于它专为边缘AI和计算机视觉应用优化&#xff0c;1.6 TOPS的NPU算力配合4K多媒体处理…...

如何5分钟快速部署GreaterWMS:终极开源仓库管理系统指南

如何5分钟快速部署GreaterWMS&#xff1a;终极开源仓库管理系统指南 【免费下载链接】GreaterWMS This Inventory management system is the currently Ford Asia Pacific after-sales logistics warehousing supply chain process . After I leave Ford , I start this projec…...

VS Code MCP插件权限控制实战:5步构建SBOM+OPA双引擎合规防护体系

更多请点击&#xff1a; https://intelliparadigm.com 第一章&#xff1a;VS Code MCP插件权限控制实战&#xff1a;5步构建SBOMOPA双引擎合规防护体系 VS Code 的 MCP&#xff08;Model Context Protocol&#xff09;插件在 AI 原生开发中日益关键&#xff0c;但其对本地文件…...

LFM2.5-VL-1.6B前端交互设计:JavaScript实现实时图像上传与结果展示

LFM2.5-VL-1.6B前端交互设计&#xff1a;JavaScript实现实时图像上传与结果展示 1. 引言&#xff1a;当AI视觉遇上Web交互 想象这样一个场景&#xff1a;用户随手拍下一张照片上传到网页&#xff0c;几秒钟后就能获得详细的文字描述和智能问答反馈。这正是LFM2.5-VL-1.6B这类…...

国产芯片适配进度告急!MCP 2026强制认证倒计时180天,你还在用X86测试环境凑合?

更多请点击&#xff1a; https://intelliparadigm.com 第一章&#xff1a;MCP 2026强制认证政策全景解读 MCP&#xff08;Model Certification Protocol&#xff09;2026 是由全球人工智能治理联盟&#xff08;GAIG&#xff09;于2024年10月正式发布的下一代大模型合规性认证框…...

人脸伪造判别分类网络CNNTransformer

一、中间件是啥&#xff1f;咱用“餐厅”打个比方 想象一下&#xff0c;你的FastAPI应用是个高级餐厅。 ?? 顾客&#xff08;客户端请求&#xff09;来到门口。- 迎宾&#xff08;CORS中间件&#xff09;&#xff1a;先看你是不是从允许的街区&#xff08;域名&#xff09;来…...

WaveDrom技术深度解析:JavaScript驱动的数字时序图渲染引擎架构与实践

WaveDrom技术深度解析&#xff1a;JavaScript驱动的数字时序图渲染引擎架构与实践 【免费下载链接】wavedrom :ocean: Digital timing diagram rendering engine 项目地址: https://gitcode.com/gh_mirrors/wa/wavedrom 项目核心价值定位&#xff1a;代码即设计的数字时…...

2025届毕业生推荐的降AI率神器实测分析

Ai论文网站排名&#xff08;开题报告、文献综述、降aigc率、降重综合对比&#xff09; TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 想要把文本被人工智能检测工具识别的概率给降下来&#xff0c;就得要从词汇多元化以及句式复…...