【Java】最新版本SpringCloudStream整合RocketMQ实现单项目中事件的发布与监听
创始人
2024-05-30 19:13:45
0

文章目录

  • 前言
  • 依赖配置
  • 代码

参考

前言

SpringCloud项目中整合RocketMQ是为了削峰填谷。
这里我使用RocketMQ的作用用于接收项目中产生的消息,然后异步的发送邮件给客户,这是这个项目的产生的背景。

依赖配置

com.alibaba.cloudspring-cloud-starter-bus-rocketmqcom.alibaba.cloudspring-cloud-starter-stream-rocketmqorg.apache.rocketmqrocketmq-clientorg.apache.rocketmqrocketmq-aclcom.alibaba.cloudspring-cloud-starter-bus-rocketmqorg.apache.rocketmqrocketmq-client4.9.4org.apache.rocketmqrocketmq-acl4.9.4

项目导入上面依赖之后即可开始代码的编写

代码

然后让我们先看一眼配置文件

# Tomcat
server:port: 9201# Spring
spring:application:# 应用名称name: towelove-systemprofiles:# 环境配置active: devcloud:nacos:discovery:# 服务注册地址server-addr: localhost:8848config:# 配置中心地址server-addr: localhost:8848# 配置文件格式file-extension: yaml# 共享配置shared-configs[0]:data-id: towelove-base-dev.yamlrefresh: trueshared-configs[1]:data-id: towelove-mysql-dev.yamlrefresh: trueshared-configs[2]:data-id: towelove-redis-dev.yamlrefresh: true# Spring Cloud Stream 配置项,对应 BindingServiceProperties 类stream:function:definition: mailSendConsumer;sendSmsToAdmin;sendSmsToUser; # 需要确保消费者类的名称和这里一样# Binding 配置项,对应 BindingProperties Mapbindings:sendSmsToAdmin-out-0: # 配置生产者destination: admin_sms_sendsendSmsToAdmin-in-0:destination: admin_sms_sendgroup: system_sms_send_consumer_groupsendSmsToUser-out-0: # 配置生产者destination: admin_sms_sendsendSmsToUser-in-0:destination: admin_sms_sendgroup: system_sms_send_consumer_group
#        smsSendConsumer-in-0: # 配置消费者
#          destination: admin_sms_send
#          group: system_sms_send_consumer_group#        smsSend-out-1:
#          destination: user_sms_send
#        smsSendConsumer-in-1:
#          destination: user_sms_send
#          group: system_sms_send_consumer_groupmailSend-out-0:destination: system_mail_sendmailSendConsumer-in-0: # 需要确保消费者类的名称和这里一样destination: system_mail_sendgroup: system_mail_send_consumer_group# Spring Cloud Stream RocketMQ 配置项rocketmq:# RocketMQ Binder 配置项,对应 RocketMQBinderConfigurationProperties 类binder:name-server: 192.168.146.115:9876 # RocketMQ Namesrv 地址#          access-key: # 用户名#          secret-key:  # 密码default: # 默认 bindings 全局配置producer: # RocketMQ Producer 配置项,对应 RocketMQProducerProperties 类group: system_producer_group # 生产者分组send-type: SYNC # 发送模式,SYNC 同步# 如果你项目里只对接一个中间件,那么不用定义binders# 当系统要定义多个不同消息中间件的时候,使用binders定义#      binders:#        my-rocketmq:#          type: rocketmq#          environment:#            rocketmq:#              name-server: 192.168.146.115:9876#          access-key: # 用户名#          secret-key:  # 密码# Spring Cloud Bus 配置项,对应 BusProperties 类bus:enabled: true # 是否开启,默认为 trueid: ${spring.application.name}:${server.port} # 编号,Spring Cloud Alibaba 建议使用“应用:端口”的格式destination: springCloudBus # 目标消息队列,默认为 springCloudBus

这里我截取了比较重要的配置,然后下面进行配置的讲解
在这里插入图片描述
首先就是我写了特别多注释的一个spring.cloud.stream.function.definition
这个东西是什么作用呢?
我的理解是,它用来声明你当前项目中的消费者,以及消费者类中的方法。
然后就是spring.cloud.stream.bindings中的好多个xxx-out-0和xxx-in-0
其中out对应的项目的输出,也就是消息的产生,对应的就是项目中的生产者,生产者发送消息的需要指定对应的信道,也就是你要告诉他往哪里发,其实就是对应的broker(再RocketMQ里面是这样子的),并且设定你发往的这个broker对应的topic,也就是destination。
那么同理,当生产者吧消息发送到broker中对应的topic后,我们就需要消费者去消费这个消息了。
那么此时就是使用in标签。
in标签里面的destination表示的也就是当前消费者需要去消费哪一个topic里面的消息。
你可能有一个疑问就是,那么为什么不用去指定对应的broker呢?
下面就是讲解这个in和out标签的声明的规则。
其实这也是一种约定优于配置的思想。
其中functionName就是你的消费者的类名或者你要提供消费的方法。
在命名规则的最后还有一个 index,它是 input 和 output 的序列,如果同一个 function name 只有一个 output 和一个 input,那么这个 index 永远都是 0。而如果你需要为一个 function 添加多个 input 和 output,就需要使用 index 变量来区分每个生产者消费者了。
Input 信道(消费者):< functionName > - in - < index >;

Output 信道(生产者):< functionName > - out - < index >。

讲解完这些,你大概就理解了这里的代码是为什么这么编写了。
那么下面我引入具体的业务代码。
我们从底层向上。
首先是消息的实体类。


@Data
public class SmsSendMessage {/*** 邮件日志编号*/@NotNull(message = "邮件日志编号不能为空")private Long logId;/*** 接收邮件地址*/@NotNull(message = "电话号码不能为空")private String phonenumber;/*** 邮件账号编号*/@NotNull(message = "邮件账号编号不能为空")private Long accountId;/*** 邮件发件人*/private String nickname;/*** 邮件标题*/@NotEmpty(message = "邮件标题不能为空")private String title;/*** 邮件内容*/@NotEmpty(message = "邮件内容不能为空")private String content;private Boolean isHtml;private File[] files;}

这个是消息的生产者

@Slf4j
@Service
public class SmsProducer {@Autowiredprivate StreamBridge streamBridge;public void sendSmsToAdmin(SmsSendMessage message) {log.info("要发送的短信内容为: {}", message);streamBridge.send("sendSmsToAdmin-out-0", message);}public void sendSmsToUser(Long userId,Long accountId) {log.info("要发送的短信内容为: {}", "userId:"+userId+"accountId:"+accountId);streamBridge.send("sendSmsToUser-out-0",  "userId:"+userId+"  accountId:"+accountId);}}

然后就是控制层

@RestController
@RequestMapping("/sys/sms")
public class SmsController {@Autowiredprivate SmsProducer smsProducer;@PostMapping("/send/admin")public R sendSmsToAdmin(@RequestBody @Valid SmsSendMessage message){smsProducer.sendSmsToAdmin(message);return R.ok();}@PostMapping("/send/user")public R sendSmsToUser(@RequestParam("userId")Long userId,@RequestParam("accountId")Long accountId){smsProducer.sendSmsToUser(userId,accountId);return R.ok();}}

然后下面是事件消费者的第一种写法

@Component
@Slf4j
public class SmsSendConsumer //implements Consumer
{//@Override//public void accept(SmsSendMessage message) {//    System.out.println(message);//}@Beanpublic Consumer sendSmsToAdmin() {return reqest -> {log.info("received: {} ", reqest);};}@Beanpublic Consumer sendSmsToUser(){return request -> {log.info("received: {}", request);List params = Arrays.stream(request.split(",")).map(Long::valueOf).collect(Collectors.toList());System.out.println(params);};}}

简单的介绍一下代码的逻辑,
其实就是我们向控制层发送一个请求并且携带上一些参数之后,控制层让生产者发送一个消息到对应的消息队列中。
发现了吗,这里消息的生产者发送的消息的目的地,就是我们设定的out标签。
在这里插入图片描述
那么消费者如何知道要去消费消息呢?
这就是为什么上面我说function.definition和in标签的作用了。
in标签这里的前缀就是我们的方法名,也就是对应的broker中的topic有消息后,对应的消费者会把消息拉过来,然后进行消费,而他之所以能知道要去消费哪一个消息也就是因为这里的绑定好的原因。
所以如果你一个类中声明了多个的消费方法,只需要再function.definition这个地方声明出你方法的名称,并且再代码里面使用@Bean的方式去声明出对应的方法即可
也就是如下图一样。
在这里插入图片描述
那么好奇的你可能会发现,这样子可以定义多个方法,还挺不错的,就是好像有点麻烦欸,要写的东西一下子就多了。
所以,如果你的消费者类只有一个方法,也就是你当前要消费的消费者只需要提供唯一的方法,那么我们可以把function.definition这里的方法名编写为消费者类的名称。
也就是下面这种代码的方式
在这里插入图片描述
而我们的生产者还是一样,只要确保其发送消息的信道是确定的即可
在这里插入图片描述
那么以这两种方式,如果你的消费者需要提供多个方法,那么就使用第一种方式,而如果你的消费者是单一的,只需要提供某一种方法,那么直接使用第二种方法去实现某个类即可。

当然,两种方式可以混合在一起实现
如果你在你的代码中出现了下图的问题
在这里插入图片描述
可以查看我下面这篇文章

解决上图的问题

类似的springcloudstream整合rocketmq的问题可以私信我一起研究

相关内容

热门资讯

按键系统安装方法 一键安装,轻松掌握按键系统安装方法准备工作在开始安装按键系统之前,我们需要做好以下准备工作: 确保...
安装新风系统的吊顶设计,吊顶设... 吊顶设计在安装新风系统中的重要性随着人们对室内空气质量要求的提高,新风系统已经成为现代家居装修的重要...
安卓系统安装苹果字体,轻松实现... 安卓系统安装苹果字体:轻松实现苹果风格随着智能手机的普及,越来越多的用户开始追求个性化的手机使用体验...
北京火灾监测系统安装,保障城市... 北京火灾监测系统安装:保障城市安全的重要举措随着城市化进程的加快,火灾事故的预防和控制成为城市安全管...
保护软件系统安装,确保数据安全... 全面解析保护软件系统安装:确保数据安全与系统稳定随着信息技术的飞速发展,保护软件系统的重要性日益凸显...
安装系统专业词语,安装系统专业... 安装系统专业术语解析随着信息技术的飞速发展,系统安装和维护已成为许多行业的基础工作。为了帮助读者更好...
报表系统门窗安装流程 系统门窗安装流程详解 一、准备工作1. 设计图纸审核:首先,仔细审核设计图纸,确保门窗的尺寸、类型和...
宝塔服务器安装手机系统,宝塔服... 宝塔服务器安装手机系统全攻略随着云计算技术的不断发展,越来越多的企业和个人开始使用宝塔服务器进行网站...
安装智能停车系统的请示,关于安... 关于安装智能停车系统的请示报告尊敬的[上级领导/部门名称]:一、背景及必要性随着城市化进程的加快,城...
白银智慧停车系统安装,打造高效... 白银智慧停车系统安装:打造高效便捷的停车新体验随着白银市汽车保有量的持续增长,停车难问题日益凸显。为...
清理电脑安装系统,电脑系统安装... 电脑系统安装与清理全攻略随着电脑使用时间的增长,系统文件和临时文件会逐渐占用大量磁盘空间,影响电脑运...
暗藏式木质门禁系统安装,暗藏式... 暗藏式木质门禁系统安装指南随着智能家居和安防技术的不断发展,暗藏式木质门禁系统因其美观、隐蔽和安全性...
安装系统怎么填写,安装系统时填... 安装系统时填写信息的详细指南在安装操作系统时,填写相关信息是必不可少的步骤。这些信息不仅有助于系统识...
清华同方超翔系统安装,清华同方... 清华同方超翔系列电脑系统安装指南一、准备工作在进行系统安装之前,我们需要做好以下准备工作: 备份重...
北海制冷系统安装,北海制冷系统... 北海制冷系统安装——打造舒适凉爽的家居环境随着夏季的到来,气温逐渐升高,制冷系统的重要性愈发凸显。在...
安卓7.0.0系统安装,轻松升... 安卓7.0.0系统安装指南:轻松升级,体验新功能一、确认手机硬件支持在开始安装安卓7.0.0系统之前...
白城隔音系统门窗安装,白城隔音... 白城隔音系统门窗安装,打造宁静家居环境随着城市化进程的加快,噪音污染已经成为影响人们生活质量的重要因...
驱动安装系统步骤图解,驱动安装... 驱动安装系统步骤图解——轻松解决设备兼容性问题一、准备工作在开始安装驱动之前,我们需要做好以下准备工...
宝山区安装新风系统,宝山区积极... 宝山区积极推进新风系统安装,打造绿色健康居住环境随着城市化进程的加快,空气质量问题日益受到关注。为了...
安装新的系统需要什么,安装新系... 安装新系统前需要准备的事项1. 确定操作系统版本首先,您需要确定要安装的操作系统版本。是Window...