Redis实现发布/订阅功能(实战篇)

前言

博主在学习 Redis 实现发布订阅功能的时候,踩了太多的坑。

不是讲解不详细,看的一知半解;就是代码有问题,实际压根跑不起来!

于是博主萌生了自己写一个最新版且全程无错的博客供各位参考。希望各位不要把我才过的坑再踩一遍。(实战篇的所有代码均由本人测试,全程无Bug。)

废话不多说,让我们进入实战篇的学习!

在开始实战篇的之前,我们先一起回顾下原理篇的内容。

Redis 发布/订阅的优点

  • 高性能:Redis 作为内存存储,具备极高的读写性能,能够快速处理发布和订阅消息。
  • 简单易用:Redis 的发布/订阅接口简单,易于集成和使用。
  • 实时性强:发布的消息会立即传递给所有订阅者,具备高实时性。

Redis 发布/订阅的缺点

  • 消息丢失:由于 Redis 是内存存储,如果 Redis 实例宕机,未处理的消息可能会丢失。
  • 无法持久化:Redis 的发布/订阅模式不支持消息持久化,无法存储和检索历史消息。
  • 订阅者不可控:发布者无法控制订阅者的数量和状态,无法保证所有订阅者都能接收到消息。
  • 无确认机制:发布者无法确认消息是否被订阅者接收和处理。

正如上述中Redis的缺点,Redis的发布订阅功能并不可靠,如果我们需要保证消息的可靠性、包括确认、重试等要求,我们还是要选择使用MQ实现发布订阅。

Redis的发布/订阅应用场景:

  • 对于消息处理可靠性要求不强
  • 消息无需持久化
  • 消费能力无需通过增加消费方进行增强
  • 架构简单 中小型系统不希望应用过多中间件

Redis发布订阅命令

命令

描述

Redis Unsubscribe 命令

指退订给定的频道

Redis Subscribe 命令

订阅给定的一个或多个频道的信息

Redis Pubsub 命令

查看订阅与发布系统状态

Redis Punsubscribe 命令

退订所有给定模式的频道

Redis Publish 命令

将信息发送到指定的频道

Redis Psubscribe 命令

订阅一个或多个符合给定模式的频道


正式开始

一、添加依赖

首先,确保你已经安装并配置好 Redis 服务器,并创建了 Spring Boot 项目,在pom.xml中引入依赖。

<!-- 所需依赖 --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><optional>true</optional></dependency>

二、配置文件中配置Redis

spring:# 项目名称application:name: test-redis-boot# Redis 配置data:redis:host: 填写自己的主机IPport: 8000password: 有则填,没有去掉这个属性database: 1# 连接超时时间timeout: 10slettuce:pool:# 连接池中的最小空闲连接min-idle: 5# 连接池中的最大空闲连接max-idle: 8# 连接池的最大数据库连接数max-active: 20# #连接池最大阻塞等待时间(使用负值表示没有限制)max-wait: -1ms

三、创建Redis配置类

创建一个配置类,用于配置 Redis连接工厂和消息监听器监听通道信息

注:此配置类无需死记硬背。只需大致了解每个方法的作用即可。

/*** @Description Redis 配置类,用于配置 Redis 连接工厂和消息监听器监听通道信息* @Author gongming.Zhang* @Date 2024/9/11 18:27* @Version 1.0*/
@Configuration
@Slf4j
public class RedisConfig {/*** 自定义 RedisTemplate 序列化方式** @param redisConnectionFactory Redis 连接的线程安全工厂* @return 模板类*/@Beanpublic RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) {RedisTemplate<Object, Object> redisTemplate = new RedisTemplate<>();// 绑定 RedisConnectionFactoryredisTemplate.setConnectionFactory(redisConnectionFactory);// 创建 Jackson2JsonRedisSerializer 序列方式,对象类型使用 Object 类型。ObjectMapper objectMapper = new ObjectMapper();objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);objectMapper.activateDefaultTyping(new LaissezFaireSubTypeValidator(), ObjectMapper.DefaultTyping.NON_FINAL);Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(objectMapper, Object.class);// 设置 RedisTemplate 序列化规则,因为 key 通常是普通的字符串,所以使用StringRedisSerializer即可,而 value 是对象时,才需要使用序列化与反序列化。redisTemplate.setKeySerializer(new StringRedisSerializer());redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);// hash key 序列化规则redisTemplate.setHashKeySerializer(new StringRedisSerializer());redisTemplate.setHashValueSerializer(jackson2JsonRedisSerializer);// 属性设置后操作redisTemplate.afterPropertiesSet();log.info("RedisTemplate 自定义序列化配置完毕...");return redisTemplate;}/*** 配置主题订阅* * 可以添加多个监听器,监听多个通道,只需要将消息监听器与订阅的通道/主题绑定即可。* addMessageListener(MessageListener listener, Collection<? extends Topic> topics):将消息监听器与多个订阅的通道/主题绑定* addMessageListener(MessageListener listener, Topic topic):将消息监听器与订阅的通道/主题绑定** @param connectionFactory Redis 连接的线程安全工厂* @return 容器对象*/@Beanpublic RedisMessageListenerContainer container(RedisConnectionFactory connectionFactory,MessageListenerAdapter listenerAdapter) {RedisMessageListenerContainer container = new RedisMessageListenerContainer();// 设置连接工厂,RedisConnectionFactory 可以直接从容器中取,也可以从 RedisTemplate 中取container.setConnectionFactory(connectionFactory);// 订阅名称叫 cache 的通道, 类似 Redis 中的 subscribe 命令container.addMessageListener(listenerAdapter, new ChannelTopic("cache"));// 订阅名称以 'test-' 开头的全部通道, 类似 Redis 的 pSubscribe 命令container.addMessageListener(listenerAdapter, new PatternTopic("test-*"));log.info("消息监听器和通道绑定完毕...");return container;}/*** 配置消息监听适配器** @param redisReceiveListener* @return*/@Beanpublic MessageListenerAdapter listenerAdapter(RedisReceiveListener redisReceiveListener) {return new MessageListenerAdapter(redisReceiveListener);}
}

四、创建消息发送服务端

创建一个消息发布类,用于向客户端发布消息。

/*** @Description 消息发布服务端* @Author gongming.Zhang* @Date 2024/9/12 9:42* @Version 1.0*/
@Component
@Slf4j
public class MessagePublisher {@Autowiredprivate RedisTemplate<Object, Object> redisTemplate;/*** 服务端发布消息** @param channel 通道名* @param message 待发送的消息*/public void sendMessage(String channel, String message) {redisTemplate.convertAndSend(channel, message);log.info("消息发送成功...  channel={}, message={}", channel, message);}}

五、创建监听器(客户端)

用于监听服务端发送的消息,每次服务端发送新消息时,都会自动调用onMessage()方法。

/*** @Description Redis 消息监听器* @Author gongming.Zhang* @Date 2024/9/11 18:53* @Version 1.0*/
/*当收到订阅的消息时,会将消息交给这个类处理。* 可以直接实现 MessageListener 接口,也可以继承它的实现类 MessageListenerAdapter.* 自动多线程处理,打印日志即可看出,即使手动延迟,也不会影响后面消息的接收。*/
@Component
@Slf4j
public class RedisReceiveListener implements MessageListener {@Autowiredprivate RedisTemplate<Object, Object> redisTemplate;/*** 处理回调逻辑。每次新消息到达时,都会调用此方法。通过 onMessage 方法执行业务代码* <p>* 该接口不仅可以访问实际消息,还可以访问接收消息的频道(Channel),以及订阅时用于匹配频道(Channel)的模式。* 此信息使被调用者不仅可以通过内容区分各种消息,还可以检查其他详细信息。** @param message 消息对象,不能为 null* @param pattern 与通道匹配的模式(如果指定),可以为 null*/@Overridepublic void onMessage(Message message, byte[] pattern) {// 1.获取消息所属的通道  -->  首先获取 字符串序列化器,再从给定的二进制数据中反序列化对象。String channel = redisTemplate.getStringSerializer().deserialize(message.getChannel());// 2.获取客户端发送的消息内容  -->  后期可以根据自己项目中 消息 的类型,来确定用什么序列化器Object msg = redisTemplate.getValueSerializer().deserialize(message.getBody());log.info("收到Redis订阅消息: channel={} msg={}", channel, msg);}
}

六、编写Controller测试

/*** @Description 测试订阅发布功能* @Author gongming.Zhang* @Date 2024/9/12 10:13* @Version 1.0*/
@RestController
@RequestMapping(value = "/api/v1/publish")
public class PublisherController {@Autowiredprivate MessagePublisher publisher;@GetMapping("/sendMessage")public String sendMessage(@RequestParam(value = "message") String message, @RequestParam(value = "channel") String channel) {publisher.sendMessage(channel, message);return "Message published: " + message;}
}

七、测试响应数据

至此,我们 SpringBoot 整合 Redis 实现发布订阅功能已经完成!


总结

通过本文,我们详细介绍了如何在 SpringBoot 中整合 Redis 实现发布/订阅功能,并提供了详细的代码示例。Redis 发布/订阅模式以其高性能和简单易用的特点,在实时消息传递场景中有着广泛的应用,但同时也需要注意其消息丢失和无法持久化等缺点,需要根据实际业务需求选择。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/425914.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

深入理解IP地址分类及子网划分详解

在互联网时代&#xff0c;IP地址是网络通信的基础。无论是访问网站、发送电子邮件&#xff0c;还是进行数据传输&#xff0c;IP地址都扮演着至关重要的角色。本文将详细解析IP地址的分类及子网划分的原理&#xff0c;帮助你更好地理解网络架构及其应用。 一、什么是IP地址 IP…

教师薪酬管理系统的设计与实现

摘 要 传统信息的管理大部分依赖于管理人员的手工登记与管理&#xff0c;然而&#xff0c;随着近些年信息技术的迅猛发展&#xff0c;让许多比较老套的信息管理模式进行了更新迭代&#xff0c;老师信息因为其管理内容繁杂&#xff0c;管理数量繁多导致手工进行处理不能满足广…

Android MediaPlayer + GLSurfaceView 播放视频

Android使用OpenGL 播放视频 概述TextureView的优缺点OpenGL的优缺点 实现复杂图形效果的场景参考 概述 在Android开发中&#xff0c;使用OpenGL ES来渲染视频是一种常见的需求&#xff0c;尤其是在需要实现自定义的视频播放界面或者视频特效时。结合MediaPlayer&#xff0c;我…

吸浮毛宠物空气净化器推荐,希喂、小米、有哈宠物空气净化器测评

养猫需谨慎&#xff0c;不然就要做猫奴一辈子啦&#xff01;上次堂妹来我家住几天&#xff0c;刚开始还担心和猫处不来&#xff0c;不敢去摸它&#xff0c;走的时候已经约好下次来看它的时间&#xff0c;笑死我了。毕竟猫咪这么可爱&#xff0c;很少有人可以抵抗它的魅力。 这不…

什么是科技与艺术相结合的异形创意圆形(饼/盘)LED显示屏

在当今数字化与创意并重的时代&#xff0c;科技与艺术的融合已成为推动社会进步与文化创新的重要力量。其中&#xff0c;晶锐创显异形创意圆形LED显示屏作为这一趋势下的杰出代表&#xff0c;不仅打破了传统显示设备的形态束缚&#xff0c;更以其独特的造型、卓越的显示效果和广…

3谐振功率放大器的实际电路设计

1原理电路 下图是谐振功率放大器的原理电路&#xff0c;如果我们照着下图搭一个电路&#xff0c;会发现它可能实现不了功率放大?这是为什么&#xff1f; 2实际电路设计 2.1要注意直流馈电线路 馈电原则(馈电供电)&#xff1a; 1&#xff09;保证直流电流分量流过直流电源&…

基于SpringBoot+Vue的校园礼服装租赁系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、SSM项目源码 系统展示 【2025最新】基于JavaSpringBootVueMySQL的…

【Java 优选算法】双指针(下)

欢迎关注个人主页&#xff1a;逸狼 创造不易&#xff0c;可以点点赞吗~ 如有错误&#xff0c;欢迎指出~ 有效三角形的个数 题目链接 解法 解法1:暴力枚举--->O(n^3) 解法2:利用单调性,使用双指针来解决---->O(n^2) 优化:对整个数组进行排序先固定最大数在最大数的左…

【模板代码的组织结构与模板的显式实例化和声明】模板代码的组织结构与模板的显式实例化和声明

一、模板的组织结构 之前对于模板&#xff0c;我们都是写在同一个 . c p p .cpp .cpp文件下&#xff0c;那如果我们将模板分开&#xff0c;单独开一个 . h .h .h和 . c p p .cpp .cpp来创建模板&#xff0c;会发生什么&#xff1f; 首先&#xff0c;我们创建一个 m y c l a s…

【2024.08】图模互补:知识图谱与大模型融合综述-笔记

阅读目的&#xff1a;假设已有一个知识图谱&#xff0c;如何利用图谱增强模型的问答&#xff0c;如何检索知识图谱、知识图谱与模型的文本如何相互交互、如何利用知识图谱增强模型回答的可解释性。 从综述中抽取感兴趣的论文进一步阅读。 来源&#xff1a;图模互补&#xff1…

阿里云 Quick BI使用介绍

Quick BI使用介绍 文章目录 阿里云 Quick BI使用介绍1. 创建自己的quick bi服务器2. 新建数据源3. 上传文件和 使用4. 开始分析 -选仪表盘5. 提供的图表6. 一个图表的设置使用小结 阿里云 Quick BI使用介绍 Quick BI是一款全场景数据消费式的BI平台&#xff0c;秉承全场景消费…

【梯度下降|链式法则】卷积神经网络中的参数是如何传输和更新的?

【梯度下降|链式法则】卷积神经网络中的参数是如何传输和更新的&#xff1f; 【梯度下降|链式法则】卷积神经网络中的参数是如何传输和更新的&#xff1f; 文章目录 【梯度下降|链式法则】卷积神经网络中的参数是如何传输和更新的&#xff1f;1. 什么是梯度&#xff1f;2.梯度…

华为eNSP使用详解

eNSP&#xff08;Enterprise Network Simulation Platform&#xff09;是华为提供的一款网络仿真平台&#xff0c;它允许用户在没有真实设备的情况下进行网络实验和学习网络技术。eNSP可以模拟各种网络设备&#xff0c;如交换机、路由器、防火墙等&#xff0c;并支持创建多种网…

Linux 环境打包 QT 程序

目录 前言 一、安装Qt库和编译器 二、编译生成可执行文件 三、准备脚本 四、添加程序依赖库 五、添加 Qt 相关依赖库 六、发布 deb 安装包 参考文章 前言 Qt是一个跨平台的GUI框架&#xff0c;可以在多种操作系统上运行&#xff0c;包括Linux、Windows和macOS。因此&am…

ASP.NET Core高效管理字符串集合

我们在开发 Web 项目时经常遇到需要管理各种来源的字符串集合&#xff08;例如HTTP 标头、查询字符串、设置的值等&#xff09;的情况。合理的管理这些字符串集合不仅可以减少出bug的几率&#xff0c;也能提高应用程序的性能。ASP.NET Core 为我们提供了一种特殊的只读结构体 S…

C/C++实现植物大战僵尸(PVZ)(打地鼠版)

&#x1f680;欢迎互三&#x1f449;&#xff1a;程序猿方梓燚 &#x1f48e;&#x1f48e; &#x1f680;关注博主&#xff0c;后期持续更新系列文章 &#x1f680;如果有错误感谢请大家批评指出&#xff0c;及时修改 &#x1f680;感谢大家点赞&#x1f44d;收藏⭐评论✍ 游戏…

【数据结构】字符串与JSON字符串、JSON字符串及相应数据结构(如对象与数组)之间的相互转换

前言&#xff1a; 下面打印日志用的是FastJSON依赖库中的 Log4j2。依赖&#xff1a; <!-- Alibaba Fastjson --> <dependency><groupId>com.alibaba</groupId><artifactId>fastjson</artifactId><version>1.2.80</version> …

[Redis] Redis中的set和zset类型

&#x1f338;个人主页:https://blog.csdn.net/2301_80050796?spm1000.2115.3001.5343 &#x1f3f5;️热门专栏: &#x1f9ca; Java基本语法(97平均质量分)https://blog.csdn.net/2301_80050796/category_12615970.html?spm1001.2014.3001.5482 &#x1f355; Collection与…

OpenCV运动分析和目标跟踪(1)累积操作函数accumulate()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 将一个图像添加到累积图像中。 该函数将 src 或其部分元素添加到 dst 中&#xff1a; dst ( x , y ) ← dst ( x , y ) src ( x , y ) if mask…

mysql笔记8(多表查询)

文章目录 1. union联合查询可能会用到去重操作 2. inner join 内连接3. left join 左连接4. right join 右连接5. cross join 交叉连接6. natural join 自然连接natural left join 自然左连接natural right join 自然右连接自然连接的两张表没有同名字段怎么办&#xff1f; 7. …