环境:SpringBoot + jdk1.8

基础配置参考
https://blog.csdn.net/llll234/article/details/80966952

查看了基础配置那么会遇到一下几个问题:

1.实际应用中可能会订阅多个通道,而一下这种写法不太通用
container.addMessageListener(listenerAdapter(new RedisPmpSub()),new PatternTopic("pmp"));

2.使用过程中使用new RedisPmpSub()配置消息接收对象会有问题。
如果RedisPmpSub既是消息接收类,也是消息处理类。那么如果此时需要注入Bean,会成功吗?

3.考虑后期的扩展性是否能尽量不改变原有代码的基础上,进行扩展

额外的配置文件

<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency> <dependency>
<groupId>commons-lang</groupId>
<artifactId>commons-lang</artifactId>
<version>RELEASE</version>
</dependency> <dependency>
<groupId>com.google.code.gson</groupId>
<artifactId>gson</artifactId>
</dependency>

由于GsonUtil依赖的是某个SDK,GsonUtil.toJson(this, BasePubMessage.class)可替换为
new Gson().toJson(this, BasePubMessage.class);
lombok需要下载插件

发布者

枚举定义

考虑到可维护性,采用枚举的方式定义管道RedisChannelEnums

 public enum RedisChannelEnums {

     /**redis频道code定义 需要与发布者一致*/
LIVE_INFO_CHANGE("LIVE_INFO_CHANGE","直播信息改变"), ;
/** 枚举定义+描述 */
private String code;
private String description; RedisChannelEnums(String code, String description) {
this.code = code;
this.description = description;
} /** 根据code获取对应的枚举对象 */
public static RedisChannelEnums getEnum(String code) {
RedisChannelEnums[] values = RedisChannelEnums.values();
if (null != code && values.length > 0) {
for (RedisChannelEnums value : values) {
if (value.code == code) {
return value;
}
}
}
return null;
} /** 该code在枚举列表code属性是否存在 */
public static boolean containsCode(String code) {
RedisChannelEnums anEnum = getEnum(code);
return anEnum != null;
} /** 判断code与枚举中的code是否相同 */
public static boolean equals(String code, RedisChannelEnums calendarSourceEnum) {
return calendarSourceEnum.code == code;
} public String getCode() {
return code;
} public String getDescription() {
return description;
} }

消息模板

为了兼容不同的业务场景,需要定义消息模板对象BasePubMessage
其中ToString方法的作用是将对象转成Json字符

 @Data
public abstract class BasePubMessage { /**发布订阅频道名称*/
protected String channel; protected String extra; @Override
public String toString() {
return GsonUtil.toJson(this, BasePubMessage.class);
} }

消息对象LiveChangeMessage
其中ToString方法的作用是将对象转成Json字符

 @Data
public class LiveChangeMessage extends BasePubMessage { /**直播Ids*/
private String liveIds; @Override
public String toString() {
return GsonUtil.toJson(this, LiveChangeMessage.class);
} }

发布者服务

public interface RedisPub {

    /**
* 集成redis实现消息发布订阅模式-双通道
* @param redisChannelEnums 枚举定义
* @param basePubMessage 消息
*/
void sendMessage(RedisChannelEnums redisChannelEnums, BasePubMessage basePubMessage); }
 @Service
public class RedisPubImpl implements RedisPub { @Resource
private StringRedisTemplate stringRedisTemplate; @Override
public void sendMessage(RedisChannelEnums redisChannelEnums, BasePubMessage basePubMessage) { if(redisChannelEnums ==null || basePubMessage ==null){
return;
} basePubMessage.setChannel(redisChannelEnums.getCode());
stringRedisTemplate.convertAndSend(redisChannelEnums.getCode(), basePubMessage.toString());
System.out.println("发布成功!");
}
}

订阅者

注解配置

RedisConfig作为订阅者的配置类,主要作用是:Redis消息监听器容器、配置消息接收处理类
同时新加入的功能解决了我们上面提出的几个问题

 @Service
@Configuration
@EnableCaching
public class RedisConfig { /**
* 存放策略实例
* classInstanceMap : key-beanName value-对应的策略实现
*/
private ConcurrentHashMap<String, BaseSub> classInstanceMap = new ConcurrentHashMap<>(20); /**
* 注入所有实现了Strategy接口的Bean
*
* @param strategyMap
* 策略集合
*/
@Autowired
public RedisConfig(Map<String, BaseSub> strategyMap) {
this.classInstanceMap.clear();
strategyMap.forEach((k, v) ->
this.classInstanceMap.put(k.toLowerCase(), v)
);
} /**
* Redis消息监听器容器
*
* @param connectionFactory
*
* @return
*/
@Bean
RedisMessageListenerContainer container(RedisConnectionFactory connectionFactory) { RedisMessageListenerContainer container = new RedisMessageListenerContainer();
container.setConnectionFactory(connectionFactory); RedisChannelEnums[] redisChannelEnums = RedisChannelEnums.values();
if (redisChannelEnums.length > 0) {
for (RedisChannelEnums redisChannelEnum : redisChannelEnums) {
if (redisChannelEnum == null || StringUtils.isEmpty(redisChannelEnum.getCode()) || redisChannelEnum.getClassName()==null) {
continue;
}
//订阅了一个叫pmp和channel 的通道,多通道
//一个订阅者接收一个频道信息,新增订阅者需要新增RedisChannelEnums定义+BaseSub的子类 String toLowerCase = redisChannelEnum.getClassName().getSimpleName().toLowerCase();
BaseSub baseSub = classInstanceMap.get(toLowerCase);
container.addMessageListener(listenerAdapter(baseSub), new PatternTopic(redisChannelEnum.getCode()));
}
}
return container;
} /**
* 配置消息接收处理类
*
* @param baseSub
* 自定义消息接收类
*
* @return MessageListenerAdapter
*/
@Bean()
@Scope("prototype")
MessageListenerAdapter listenerAdapter(BaseSub baseSub) {
//这个地方 是给messageListenerAdapter 传入一个消息接受的处理器,利用反射的方法调用“receiveMessage”
//也有好几个重载方法,这边默认调用处理器的方法 叫handleMessage 可以自己到源码里面看
//注意2个通道调用的方法都要为receiveMessage
return new MessageListenerAdapter(baseSub, "receiveMessage");
} }

  

@Autowired
public RedisConfig(Map<String, BaseSub> strategyMap) 方法的作用是将所有的配置消息接收处理类注入进来,那么消息接收处理类里面的注解对象也会注入进来。
解决了我们提出的第二个问题 而String toLowerCase = redisChannelEnum.getClassName().getSimpleName().toLowerCase();
BaseSub baseSub = classInstanceMap.get(toLowerCase);
container.addMessageListener(listenerAdapter(baseSub), new PatternTopic(redisChannelEnum.getCode()));
是根据不同的管道对应不同的订阅者,也就是一个订阅者对应一个管道。方便根据不同的业务场景进行处理。
使用这种方式主需要配置redisChannelEnum枚举即可,解决了我们提出的第一个问题。
这样一来,订阅者就变得比较通用了

  

枚举

RedisChannelEnums作用:定义不同管道对应的订阅者,后期增加一个管道类型只需要增加一个枚举即可

 public enum RedisChannelEnums {

     /**redis频道名称定义 需要与发布者一致*/
LIVE_INFO_CHANGE("LIVE_INFO_CHANGE", LiveChangeSub.class, "直播信息改变"), ;
/** 枚举定义+描述 */
private String code;
private Class<? extends BaseSub> className;
private String description; RedisChannelEnums(String code, Class<? extends BaseSub> className, String description) {
this.code = code;
this.className=className;
this.description = description;
} /** 根据code获取对应的枚举对象 */
public static RedisChannelEnums getEnum(String code) {
RedisChannelEnums[] values = RedisChannelEnums.values();
if (null != code && values.length > 0) {
for (RedisChannelEnums value : values) {
if (value.code == code) {
return value;
}
}
}
return null;
} /** 该code在枚举列表code属性是否存在 */
public static boolean containsCode(String code) {
RedisChannelEnums anEnum = getEnum(code);
return anEnum != null;
} /** 判断code与枚举中的code是否相同 */
public static boolean equals(String code, RedisChannelEnums calendarSourceEnum) {
return calendarSourceEnum.code == code;
} public String getCode() {
return code;
} public String getDescription() {
return description;
} public Class<? extends BaseSub> getClassName() {
return className;
}
}

消息模板

BaseSubMessage定义通用的字段,与json字符的通用转换

 @Data
abstract class BaseSubMessage { /** 发布订阅频道名称 */
private String channel; private String extra; private String json; BaseSubMessage(String json) {
if(StringUtils.isEmpty(json)){
return;
} this.json = json;
Map map = new Gson().fromJson(this.json, Map.class);
BeanHelper.populate(this, map);
} }

LiveChangeMessage定义当前业务场景的字段

 @Data
@ToString(callSuper = true)
public class LiveChangeMessage extends BaseSubMessage { /** 直播Ids */
private String liveIds; public LiveChangeMessage(String json) {
super(json);
} }

订阅者服务

BaseSub定义接收消息的通用方法

 public interface BaseSub {

     /**
* 接收消息
* @param jsonMessage json字符
*/
void receiveMessage(String jsonMessage);
}

LiveChangeSub具体消息接收对象

 @Component
public class LiveChangeSub implements BaseSub { /**只是定义的注解测试,可以换成自己的*/
@Autowired
private CategoryMapper categoryMapper; @Override
public void receiveMessage(String jsonMessage) { System.out.println("项目aries-server.....................");
//注意通道调用的方法名要和RedisConfig2的listenerAdapter的MessageListenerAdapter参数2相同
System.out.println("这是 LiveChangeSub" + "-----" + jsonMessage); LiveChangeMessage liveChangeMessage = new LiveChangeMessage(jsonMessage);
System.out.println(liveChangeMessage); Category category = categoryMapper.get(1L);
System.out.println("category:" + category); }
}

总结

发布者配置场景:独立的服务器,独立的项目,A redis缓存服务器
订阅者配置场景:不同于发布者的独立的服务器,独立的项目,A redis缓存服务器
使用场景:一个发布者、一个或者多个订阅者。发布者负责发布消息,订阅者负责接收消息。一旦发布者消息发布出来,那么
订阅者可以通过管道进行监听。同时可以根据不同的管道设置不同的消息接收者或者叫消息处理者。 优点:容易配置,好管理
缺点:由于基于redis去做,不同的redis服务就不适用了。需要考虑消息丢失,持久化的问题。

springboot集成redis实现消息发布订阅模式-双通道(跨多服务器)的更多相关文章

  1. Springboot+Redis(发布订阅模式)跨多服务器实战

    一:redis中发布订阅功能(http://www.redis.cn/commands.html#pubsub) PSUBSCRIBE pattern [pattern -]:订阅一个或者多个符合pa ...

  2. 15天玩转redis —— 第九篇 发布/订阅模式

    本系列已经过半了,这一篇我们来看看redis好玩的发布订阅模式,其实在很多的MQ产品中都存在这样的一个模式,我们常听到的一个例子 就是邮件订阅的场景,什么意思呢,也就是说100个人订阅了你的博客,如果 ...

  3. redis实现消息发布/订阅

    redis实现简单的消息发布/订阅模式. 消息订阅者: package org.common.component; import org.slf4j.Logger; import org.slf4j. ...

  4. redis 的消息发布订阅

    redis支持pub/sub功能(可以用于消息服务器),这个功能类似mq,这里做一个简单的介绍 Pub/Sub Pub/Sub 从字面上理解就是发布(Publish)与订阅(Subscribe),在R ...

  5. SpringBoot Redis 发布订阅模式 Pub/Sub

    SpringBoot Redis 发布订阅模式 Pub/Sub 注意:redis的发布订阅模式不可以将消息进行持久化,订阅者发生网络断开.宕机等可能导致错过消息. Redis命令行下使用发布订阅 pu ...

  6. Spring Data Redis实现消息队列——发布/订阅模式

    一般来说,消息队列有两种场景,一种是发布者订阅者模式,一种是生产者消费者模式.利用redis这两种场景的消息队列都能够实现. 定义:生产者消费者模式:生产者生产消息放到队列里,多个消费者同时监听队列, ...

  7. redis消息通知(任务队列/优先级队列/发布订阅模式)

    1.任务队列 对于发送邮件或者是复杂计算这样的操作,常常需要比较长的时间,为了不影响web应用的正常使用,避免页面显示被阻塞,常常会将此类任务存入任务队列交由专门的进程去处理. 队列最基础的方法如下: ...

  8. redis实现消息队列&发布/订阅模式使用

    在项目中用到了redis作为缓存,再学习了ActiveMq之后想着用redis实现简单的消息队列,下面做记录.   Redis的列表类型键可以用来实现队列,并且支持阻塞式读取,可以很容易的实现一个高性 ...

  9. Redis消息通知(任务队列和发布订阅模式)

    Redis学习笔记(十)消息通知(任务队列和发布订阅模式) 1. 任务队列 1.1 任务队列的特点 任务队列:顾名思义,就是“传递消息的队列”.与任务队列进行交互的实体有两类,一类是生产者(produ ...

随机推荐

  1. js常用设计模式实现(三)建造者模式

    创建型模式 创建型模式是对一个类的实例化过程进行了抽象,把对象的创建和对象的使用进行了分离 关于创建型模式,已经接近尾声了,还剩下建造者模式和原型模式,这一篇说一说建造者模式 建造者模式的定义 将一个 ...

  2. Akka-CQRS(16)- gRPC用JWT进行权限管理

    前面谈过gRPC的SSL/TLS安全机制,发现设置过程比较复杂:比如证书签名:需要服务端.客户端两头都设置等.想想实际上用JWT会更加便捷,而且更安全和功能强大,因为除JWT的加密签名之外还可以把私密 ...

  3. C++学习书籍推荐《C++程序设计原理与实践》下载

    百度云及其他网盘下载地址:点我 编辑推荐 <C++程序设计原理与实践>是经典程序设计思想与C++开发实践的完美结合,是C++之父回归校园后对C++编程原理和技巧的全新阐述.书中全面地介绍了 ...

  4. Pownerdesigner画用例图_类图_时序图

    1. 问题描述 软件过程中,设计阶段有几个常用的工具:Rational Rose.Visio.Pownerdesigner,一般用Rose用例图/类图/时序图,Visio画流程图,Pownerdesi ...

  5. NOIP 2017 惊魂记

    考完了NOIP三周后才开始补……然后又补了一周…… DAY -1: 晚上吃了一顿送行宴散伙饭,然后默默地看了一遍之前所有考试后写的题解,再读了几遍板子,然后和QTY一起和达哥又一次在外面谈了一个小时, ...

  6. BeautifulSoup库整理

    BeautifulSoup库 一.BeautifulSoup库的下载以及使用 1.下载 pip3 install beautifulsoup4 2.使用 improt bs4 二.BeautifulS ...

  7. 使用nvm管理多个不同版本的nodeJS之安装成功nodeJs之后使用npm报错的问题

    使用nvm安装nodeJS之后,node -v命令可以正常使用,但是npm命令一直报“npm不是内部命令”的错误,深入研究之后得到以下解决方案: 搭建步骤: (1)下载nvm   https://gi ...

  8. windows和linux下如何对拍

    对拍是各种计算机考试检查时必备工具,实际上十分强大,只要你的暴力没有写错就没有问题. 对拍的意思:(怎么有点语文课的意思雾) 对:看见'对'就可以知道有两个. 拍:就是把两个程序结果拍在一起,对照(有 ...

  9. 【原创】这一次,Chrome表现和IE11一样令人失望,围观群众有:Edge,Firefox

    前言 俗话说,常在河边走哪能不湿鞋,天天和浏览器打交道,发现浏览器竟然也隐藏BUG也不是新鲜事了.可以看下我之前的文章: [原创]分享IE7一个神奇的BUG(不是封闭标签的问题,的确是IE7的BUG) ...

  10. excel表数据生成定长txt数据

    项目作业中需要造数据,从txt文件中获取定长数据,直接从txt中修改,会显得十分麻烦,于是便利用excel自带的vba写了一个小工具.效果如下: A1表示字段名,A2表示长度,A3是数据,也可以增加字 ...