SpringBoot整合消息中间件(ActiveMQ,RabbitMQ,RocketMQ,Kafka)

news2025/1/23 7:24:40

消息中间件

  • 消息
    • 消息队列
      • JMS
    • AMQP
    • MQTT
    • Kafka
  • Spring整合消息队列
    • 模拟消息队列的工作流程
    • Spring整合ActiveMQ
    • Spring整合RabbitMQ
      • 直连交换机模式
      • 主题交换机模式
    • Spring整合RocketMQ
    • Spring整合kafka

消息

消息的发送方:生产者
消息的接收方:消费者
同步消息:发送方发送消息到接收方,接收方有所回应后才能够进行下一次的消息发送
异步消息:不需要接收方回应就可以进行下一步的发送

消息队列

什么是消息队列?
消息队列
当此时有很多个用户同时访问服务器,需要服务器进行操作,但此时由于操作太多服务器运转不过来,这时将非常多的操作转换成消息的格式储存器来,所有的子服务器从中获取到消息进行操作分担主服务器的压力,而这个中间存储消息的容器我们一般称为消息队列

  • 企业级应用中广泛使用的三种异步消息传递技术(实现高并发的有效处理):
  1. JMS
  2. AMQP
  3. MQTT

JMS

(java Message Service):一个规范,等同于JDBC规范,提供了与消息服务相关的API接口

  • JMS消息模型
  1. peer-2-peer: 点对点模型,消息发送到一个队列中,队列保存信息,队列的消息只能被一个消费者消费,或超时
  2. publish-subscribe:发布订阅模式,消息可以被多个消费者消费,生产者和消费者完全独立,不需要感知对方存在
  • JMS消息种类

TextMessage,MapMessage, BytesMessage,StreamMessage,ObjectMessage,Message(Message只有消息头和属性)

  • 实现JMS规范的MQ

ActiveMQ,Redis,HornetMQ,RabbitMQ,RocketMQ(RocketMQ并未完全遵守JMS规范)

AMQP

AMQP(advanced message queuing protocol):一种协议(高级队列协议,消息代理规范),规范了网络交换的数据格式,兼容JMS

JMS存在一定的问题,JMS规范对对应的语言进行了规范,但若是我使用不是规范语言进行操作的时候就会出现问题,这时我们推出AMQP,这更像是一种协议,规范消息的格式,就是无论用什么语言什么环境都无所谓,它只人消息的格式

优点:跨平台性,服务器供应商,生产者,消费者可以使用不同的语言来实现

  • AMQP的消息模型

direct exchange,fanout exchange,topic exchange,headers exchange,system exchange

AMQP的消息种类:byte[]

  • 实现AMQP的MQ:

RabbitMQ,StormMQ,RocketMQ

MQTT

(Message Queueing Telemetry Transport)消息队列遥测传输,专为小设备设计,是物联网(IOT)生态系统中主要成分之一

Kafka

kafka,一种高吞吐量的分布式订阅消息系统,提供实时消息功能

Spring整合消息队列

模拟消息队列的工作流程

模拟消息队列的处理过程

import java.util.ArrayList;

@Service
public class Messageservice implements MessageService {
    private ArrayList<String> megList=new ArrayList<String>();

    @Override
    public void sendMessage(String id) {
        System.out.println("将待发送的消息订单纳入到处理队列.id:"+id);
        megList.add(id);
    }

    @Override
    public String doMessage() {
        String remove = megList.remove(0);
        System.out.println("已完成短信业务的发送,id:"+remove);
        return remove;
    }
}

模拟将消息导入到消息队列

@Service
public class orderserviceimpl implements orderService {
    @Autowired
    private MessageService messageService;
    @Override
    public void order(String id) {
        //发送消息队列
        messageService.sendMessage(id);
    }
}

Spring整合ActiveMQ

首先下载activeMQ
下载地址:https://activemq.apache.org/components/classic/download/
下载之后进行解压缩

  • 启动服务

打开x64的bin目录下执行activemq.bat命令启动服务
在这里插入图片描述
在这里插入图片描述
启动成功 ,其中给出其web控制台的访问地址:
在这里插入图片描述
进入其管理界面:
在这里插入图片描述
默认用户名&密码:admin

  • SpringBoot进行整合activemq

添加依赖:

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-activemq</artifactId>
        </dependency>

添加配置:
配置Spirng连接的地址,以及后边消息存入的位置

server:
  port: 80
spring:
  activemq:
  # 说明spring连接的active的端口地址
    broker-url: tcp://localhost:61616

进行消息队列的操作:


@Service
public class Messageservice implements MessageService {
    @Autowired
    private JmsMessagingTemplate jmsMessagingTemplate;
    @Override
    public void sendMessage(String id) {
        System.out.println("将待发送的消息订单纳入到处理队列.id:"+id);
        jmsMessagingTemplate.convertAndSend(id);
    }
    @Override
    public String doMessage() {
        //将消息队列中的类型转移出来,并在参数中规定转移出来的消息类型
        String s = jmsMessagingTemplate.receiveAndConvert(String.class);
        System.out.println("已完成短信业务的发送,id:"+s);
        return s;
    }
}

在发送和获取期间也可以规定名称

 jmsMessagingTemplate.convertAndSend("order.shishi.id",id);
 String s = jmsMessagingTemplate.receiveAndConvert("order.shishi.id",String.class);

上述之中也有一个小问题,就是在并不是每次消费都需要进行访问,而是当消息队列中有消息就开始消费我们可以创建一个Listener


@Component
public class MessageListener {
    @JmsListener(destination = "order.shishi.id")
    public void receive(String id){
        System.out.println("已完成的短信业务:id:"+id);
    }
}

这样就自动监听指定位置下的消息,一有消息就自动开始消费,从服务开始就一直存在
还有一个消息转发的操作:

@Component
public class MessageListener {
    @JmsListener(destination = "order.shishi.id")
    @SendTo("order.bushi.id")
    public void receive(String id){
        System.out.println("已完成的短信业务:id:"+id);
    }
}

注解 @SendTo的作用是将监听到的消息消费之后将返回值返回到对应的消息中去
上述使用的都是点对点的模型,如果要使用发布订阅的模型,可以在配置文件中进行配置:

spring:
  activemq:
    broker-url: tcp://localhost:61616
  jms:
    template:
      default-destination: shishi
    pub-sub-domain: true

Spring整合RabbitMQ

rabbitMQ基于Erlang语言编写,需要安装Erlang
首先需要下载Erlang:
下载地址:https://www.erlang.org/downloads
下载完成之需要重启一下操作系统(重启电脑)
配置环境变量
在这里插入图片描述
添加path:
在这里插入图片描述
安装完成后下载RabbitMQ
下载地址:https://rabbitmq.com/install-windows.html

  • 启动rabbitMQ

在这里插入图片描述
注意:要启动rabbitMQ服务需要命令行进入到管理员身份运行
rabbitMQ的控制台界面(需要手动配置插件):
在sbin目录下找到:rabbitmq:plugins.bat命令
在这里插入图片描述
执行命令展示其插件列表,通过命令开启插件
在这里插入图片描述
这样就可以访问它的控制台界面,端口号是15672,地址:http://localhost:15672
在这里插入图片描述
输入默认密码:guest

Spring进行整合rabbitMQ首先添加依赖:

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-amqp</artifactId>
        </dependency>

在配置文件中进行rabbits的配置:

spring:
 activemq:
   broker-url: tcp://localhost:61616
 jms:
   template:
     default-destination: shishi
   pub-sub-domain: true
 rabbitmq:
   host: localhost
   port: 5672

直连交换机模式

使用直连模式的交换机进行消息队列的开发:
首先需要在配置类中进行直连交换机与消息队列的绑定


import org.springframework.amqp.core.Binding;
import org.springframework.amqp.core.BindingBuilder;
import org.springframework.amqp.core.DirectExchange;
import org.springframework.amqp.core.Queue;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class ConfigQM {
   @Bean
   public Queue directQueue(){
       //第一个是消息队列的名称,第一个true表示消息持久化,第二个表示当前的消息队列是否是连接专用(连接一关消息队列就关闭),第三个参数是是否删除(当消费者生产者都不使用就删除)
       return new Queue("direct_queue",true,true,true);
   }
   //我们需要一个交换机去绑定消息队列,此处设置一个交换机
   @Bean
   public DirectExchange directExchange(){
       return new DirectExchange("directexchange");
   }
   @Bean
   public Binding binding(){
       //将消息队列与交换机进行绑定
       return BindingBuilder.bind(directQueue()).to(directExchange()).with("direct");
   }
}

绑定之后通过直连交换机进行消息的存储

@Service
public class amqpservice implements MessageService{

   @Autowired
   private AmqpTemplate amqpTemplate;
   @Override
   public void sendMessage(String id) {
       //使用直连交换机
       amqpTemplate.convertAndSend("directExchange","direct",id);
   }
   @Override
   public String doMessage() {
       return null;
   }
}

然后从消息队列中读取消息写在rabbitMQ监听器下面:

import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;
@Component
public class RabbitMQListener {
   @RabbitListener(queues = "direct_queue")
   public void reveive(String id){
       System.out.println("已完成短信发送业务 id:"+id);
   }
}

主题交换机模式

主题交换机可以模糊设置交换机绑定的名称来达到分发的目的
例如:

    @Bean
   public Binding binding(){
       //将消息队列与交换机进行绑定
       return BindingBuilder.bind(directQueue()).to(directExchange()).with("topic_*_id");
   }

在消息进入消息队列的时候:

        amqpTemplate.convertAndSend("directExchange","topic_ni_id",id);
        amqpTemplate.convertAndSend("directExchange","topic_bu_id",id);

这个两种消息都可以进入到消息队列中去,而且通过这种方式也可以使消息进入到不同的消息队列中去

  • 绑定案件的规则:

*(星号):用来表示一个单词,且该单词必须出现
#(井号):用来表示任意数量
在这里插入图片描述

Spring整合RocketMQ

下载地址:https://rocketmq.apache.org/
默认服务端口:9876
配置环境变量:ROCKETMQ_HOME,PATH,NAMESER_ADDR(建议):127.0.0.1:9876

  • 命名服务器与broker

在这里插入图片描述
当后期的业务服务器增多时,就需要不停的进行服务器之间的连接,会变得非常繁琐,但是如果我们有一台服务器将所有的业务服务器注册进行,消费者与生产者只需要连接命名服务器即可

  • 首先启动命名服务器
    在这里插入图片描述
    双击文件启动命名服务器
    在这里插入图片描述
    然后双击mqbroker文件启动服务器:
    在这里插入图片描述
    如何测试服务器是否正常启动:
    在bin目录下启动cmd:
    在这里插入图片描述

首先使用第一个命名生成对应的消息:
在这里插入图片描述
再使用第二个命令对生成的消息进行消费:
在这里插入图片描述
进行整合:
首先导入依赖坐标:

        <dependency>
            <groupId>org.apache.rocketmq</groupId>
            <artifactId>rocketmq-spring-boot-starter</artifactId>
            <version>2.2.1</version>
        </dependency>

在配置文件中配置其命名服务器:
rocketmq是与spring在同一层次下

rocketmq:
   name-server: localhost:9876
   producer:
      group: group_rocketmq

进行消息队列的相关操作:

@Service
public class MessageRocketmqimpl implements MessageService {
   @Autowired
   private RocketMQTemplate rocketMQTemplate;
   @Override
   public void sendMessage(String id) {
       rocketMQTemplate.convertAndSend("sdasda",id);
   }
   @Override
   public String doMessage() {
       return null;
   }
}

消费者监听器:

 
@Component
@RocketMQMessageListener(topic = "sdasda",consumerGroup = "group_rocketmq")
public class MessageRocketmqListener implements RocketMQListener<String> {

    @Override
    public void onMessage(String s) {
        System.out.println("id:"+s);
        
    }
}

使用异步方式进行发送:


@Service
public class MessageRocketmqimpl implements MessageService {
    @Autowired
    private RocketMQTemplate rocketMQTemplate;
    @Override
    public void sendMessage(String id) {
        SendCallback callback=new SendCallback() {
            @Override
            public void onSuccess(SendResult sendResult) {
                System.out.println("消息发送成功");
            }

            @Override
            public void onException(Throwable throwable) {
                System.out.println("消息发送失败");
            }
        }
//        rocketMQTemplate.convertAndSend("sdasda",id);
        rocketMQTemplate.asyncSend("sdasda",id,callback);
    }
  • 同步发送与异步发送的区别:

同步发送和异步发送是两种不同的消息发送方式。在同步发送中,发送线程会等待消息发送完成并收到发送结果后继续执行,而在异步发送中,发送线程不会阻塞,可以立即执行后续逻辑。选择哪种方式取决于业务需求和对消息发送结果的要求。

Spring整合kafka

下载地址:https://kafka.apache.org/downloads
下载之后进行解压缩文件
解压之后首先需要运行:zookeeper-server-start.bat文件
这个文件相当于一个注册中心,需要先进行注册才能够启动kafka服务器,作用相当于RocketMQ中的命名服务器,需要在对应目录下cmd命令携带参数进行启动:
在这里插入图片描述
启动注册服务器后,然后开启kafka服务器:
在这里插入图片描述
在这里插入图片描述
spring进行整合kafka:
导入依赖坐标:

        <dependency>
            <groupId>org.springframework.kafka</groupId>
            <artifactId>spring-kafka</artifactId>
        </dependency>

在配置文件中进行配置,配置注册服务器的地址:

  kafka:
    bootstrap-servers: localhost:9092
    consumer:
      group-id: order
@Service
public class kafka implements MessageService {
    @Autowired
    private KafkaTemplate<String,String> kafkaTemplate;
    @Override
    public void sendMessage(String id) {
        kafkaTemplate.send("adad",id);
    }

    @Override
    public String doMessage() {
        return null;
    }
}

创建消费者监听器:


@Component
public class kafkaListener {
    @KafkaListener(topics = "adad")
    public void onMessage(ConsumerRecord<String,String> consumerRecord){
        System.out.println("id:"+consumerRecord.value());
    }
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【机器学习300问】70、向量化技术来计算神经网络时维度如何确保正确?

一、向量化技术在进行神经网络计算时的优势 向量化是一种优化技术&#xff0c;通过使用数组操作代替for循环&#xff0c;可以大大提高代码的性能和效率。在深度学习中尤其明显&#xff0c;可以提高计算效率、简化代码、优化内存使用。 二、如何确保计算时维度是正确的&#xf…

【讲解下常见的Web前端框架】

&#x1f308;个人主页: 程序员不想敲代码啊 &#x1f3c6;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f44d;点赞⭐评论⭐收藏 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共…

1260. 二维网格迁移

1260. 二维网格迁移 原题链接&#xff1a;完成情况&#xff1a;解题思路&#xff1a;参考代码&#xff1a;错误经验吸取 原题链接&#xff1a; 1260. 二维网格迁移 https://leetcode.cn/problems/shift-2d-grid/description/ 完成情况&#xff1a; 解题思路&#xff1a; 这…

【Bugku】sqli-0x1

1.打开靶场&#xff0c;进入实验场景 2.按F12查看源代码&#xff0c;发现有一个/?pls_help路径&#xff0c;在url后加上查看。 3.得到的php源码 首先&#xff0c;代码通过 error_reporting(0) 和 error_log(0) 关闭了错误报告&#xff0c;这可以防止攻击者从错误信息中获取敏…

Rust面试宝典第2题:逆序输出整数

题目 写一个方法&#xff0c;将一个整数逆序打印输出到控制台。注意&#xff1a;当输入的数字含有结尾的0时&#xff0c;输出不应带有前导的0。比如&#xff1a;123的逆序输出为321&#xff0c;8600的逆序输出为68&#xff0c;-609的逆序输出为-906。 解析 这道题本身并没有什么…

【python】python天气气候数据抓取分析可视化(源码+数据+可视化+报告)【独一无二】

&#x1f449;博__主&#x1f448;&#xff1a;米码收割机 &#x1f449;技__能&#x1f448;&#xff1a;C/Python语言 &#x1f449;公众号&#x1f448;&#xff1a;测试开发自动化【获取源码商业合作】 &#x1f449;荣__誉&#x1f448;&#xff1a;阿里云博客专家博主、5…

记录flume运行时报NullPointerException异常

【背景说明】 我要起一个将kafka上的topic_log主题中的数据上传到hdfs上的flume进程。 这是我的flume配置文件脚本&#xff1a; #定义组件 a1.sourcesr1 a1.channelsc1 a1.sinksk1#配置source1 a1.sources.r1.type org.apache.flume.source.kafka.KafkaSource a1.sources.r…

频率域滤波基础(离散傅里叶变换使用填充的缺陷)

本来是个很简单的问题&#xff0c;作者硬是写的这么复杂&#xff0c;翻译还搞错了。重点是我发现作者真正有用的东西没讲到&#xff0c;比如相位和谱如何影响图像。连个转换公式都没有&#xff0c;我只能说作者是在混字数。 首先看关于中心对称是什么意思&#xff1f;我木太明白…

从51到ARM裸机开发实验(009)LPC2138 中断实验

一、场景设计 中断的概念在《从51到ARM裸机开发实验(007) AT89C51 中断实验》中已经介绍过&#xff0c;LPC2138的Keil工程创建在《从51到ARM裸机开发实验(005)LPC2138 GPIO实验》中已经介绍过。本次使用LPC2138来实现一个这样的场景&#xff1a;四个LED依次亮灭&#xff0c;时间…

c++总结笔记(一)

计算机可以将程序转化为二进制指令&#xff08;即机器码&#xff09;&#xff0c;并由CPU执行&#xff0c;CPU会按照指令的顺序依次执行每个指令。 C语言特点&#xff1a; 简洁高效可移植模块化标准化 C语言的标准 C89(C90)标准C99标准C11标准 导入 使用include导入包含…

postgresql 备份恢复相关知识点整理归纳 —— 筑梦之路

概述 PG一般有两种备份方式&#xff1a;逻辑备份和物理备份 逻辑备份对于数据量大的场景下耗时较长&#xff0c;恢复也会耗时较长 物理备份拷贝文件的方式相对来说耗时较短&#xff0c;跟磁盘读写性能和网络传输性能有关 逻辑备份 pg_dump pg_dump 将表结构及数据以SQL语句…

Angular学习第四天--问题记录及父子组件问题

问题一、 拉取完项目&#xff0c;使用npm install命令的时候遇到的。 解决办法&#xff1a; 在查找网上五花八门的解决方案之后&#xff0c;发现都不能解决。 我的解决办法是&#xff1a; 1. 把package-lock.json给删掉&#xff1b; 2. 把package.json中公司自己库的包给删除掉…

【论文阅读】TransGNN

一、摘要 本文主要是在推荐系统中对GNN的改进。在协同过滤中&#xff0c;主要是对用户-项目交互图进行建模。但是基于GNN的方法遇到了有限的接受域和嘈杂的“兴趣无关”连接的挑战。相比之下&#xff0c;基于Transformer的方法擅长于自适应地和全局地聚合信息但是在大规模交互…

网站如果在日益变化的网络攻击中寻到一线生机

一、引言 在数字化浪潮席卷全球的今天&#xff0c;网络空间早已成为国家安全、经济发展和社会稳定的战略高地。然而&#xff0c;这片看似平静的虚拟世界&#xff0c;实则暗流涌动&#xff0c;网络攻击层出不穷&#xff0c;手段日益翻新&#xff0c;给网站的安全运营带来了前所…

Linux UDP通信系统

目录 一、socket编程接口 1、socket 常见API socket()&#xff1a;创建套接字 bind()&#xff1a;将用户设置的ip和port在内核中和我们的当前进程关联 listen() accept() 2、sockaddr结构 3、inet系列函数 二、UDP网络程序—发送消息 1、服务器udp_server.hpp initS…

物联网的核心价值是什么?——青创智通

工业物联网解决方案-工业IOT-青创智通 物联网&#xff0c;这个词汇在当今的科技领域已经变得耳熟能详。但当我们深入探索物联网的核心价值时&#xff0c;我们会发现它远不止是一个简单的技术概念&#xff0c;而是一种能够彻底改变我们生活方式和工作方式的革命性力量。 物联网…

OpenCV基本图像处理操作(五)——图像数据操作

数据读取 cv2.IMREAD_COLOR&#xff1a;彩色图像cv2.IMREAD_GRAYSCALE&#xff1a;灰度图像 import cv2 #opencv读取的格式是BGR import matplotlib.pyplot as plt import numpy as np %matplotlib inline imgcv2.imread(cat.jpg)数据显示 #图像的显示,也可以创建多个窗口 c…

P2P通信基本原理

在数字世界的脉络中&#xff0c;点对点&#xff08;P2P&#xff09;技术如同一条悄无声息的河流&#xff0c;流经信息的每个角落&#xff0c;连接着世界各地的计算机和设备。这种去中心化的网络模型&#xff0c;不仅打破了传统的客户端-服务器架构的界限&#xff0c;还赋予了数…

✌粤嵌—2024/3/11—跳跃游戏

代码实现&#xff1a; 方法一&#xff1a;递归记忆化 int path; int used[10000];bool dfs(int *nums, int numsSize) {if (path numsSize - 1) {return true;}for (int i 1; i < nums[path]; i) {if (used[path i]) {continue;}path i;used[path] 1;if (dfs(nums, num…

C++|运算符重载(1)|为什么要进行运算符重载

写在前面 本篇里面的日期类型加法&#xff0c;先不考虑闰年&#xff0c;平年的天数&#xff0c;每月的天数统一按30天算&#xff0c;那么每一年也就是360天 目录 写在前面 定义 基本数据类型 自定义数据类型 成员函数解决相加问题 Date类&#xff0b;整形 下一篇----运…