• RabbitMQ的使用


    一、消息中间件RabbitMQ---概述和概念 【一】

    1、概述

    1、大多应用中,可通过消息服务中间件来提升系统异步通信、扩展解耦能力
    2、消息服务中两个重要概念:

    • 消息代理(message broker)和目的地(destination)
      当消息发送者发送消息以后,将由消息代理接管,消息代理保证消息传递到指定目的地。

    3、消息队列主要有两种形式的目的地

    • 队列(queue):点对点消息通信(point-to-point)
    • 主题(topic):发布(publish)/订阅(subscribe)消息通信

    4、点对点式:

    • 消息发送者发送消息,消息代理将其放入一个队列中,消息接收者从队列中获取消息内容,消息读取后被移出队列
    • 消息只有唯一的发送者和接受者,但并不是说只能有一个接收者

    5、发布订阅式:

    • 发送者(发布者)发送消息到主题,多个接收者(订阅者)监听(订阅)这个主题,那么就会在消息到达时同时收到消息

    6、JMS(Java Message Service)JAVA消息服务:

    • 基于JVM消息代理的规范。ActiveMQ、HornetMQ是JMS实现

    7、 AMQP(Advanced Message Queuing Protocol)

    • 高级消息队列协议,也是一个消息代理的规范,兼容JMS
    • RabbitMQ是AMQP的实现

    2、RabbitMQ概念

    2.1 RabbitMQ简介

    RabbitMQ是一个由erlang开发的AMQP(Advanved Message Queue Protocol)的开源实现

    2.2 核心概念

    Message:消息,消息是不具名的,它由消息头和消息体组成。消息体是不透明的,而消息头则由一系列的可选属性组成,这些属性包括routing-key(路由键)、priority(相对于其他消息的优先权)、delivery-mode(指出该消息可能需要持久性存储)等。

    Publisher:消息的生产者,也是一个向交换器发布消息的客户端应用程序。

    Exchange:交换器,用来接收生产者发送的消息并将这些消息路由给服务器中的队列。Exchange有4种类型:direct(默认),fanout, topic, 和headers,不同类型的Exchange转发消息的策略有所区别

    Queue:消息队列,用来保存消息直到发送给消费者。它是消息的容器,也是消息的终点。一个消息可投入一个或多个队列。消息一直在队列里面,等待消费者连接到这个队列将其取走。

    Binding绑定,用于消息队列和交换器之间的关联。一个绑定就是基于路由键将交换器和消息队列连接起来的路由规则,所以可以将交换器理解成一个由绑定构成的路由表。Exchange 和Queue的绑定可以是多对多的关系。

    Connection:网络连接,比如一个TCP连接。

    Channel:信道,多路复用连接中的一条独立的双向数据流通道。信道是建立在真实的TCP连接内的虚拟连接,AMQP 命令都是通过信道发出去的,不管是发布消息、订阅队列还是接收消息,这些动作都是通过信道完成。因为对于操作系统来说建立和销毁 TCP 都是非常昂贵的开销,所以引入了信道的概念,以复用一条 TCP 连接。

    Consumer:消息的消费者,表示一个从消息队列中取得消息的客户端应用程序。

    Virtual Host:虚拟主机,表示一批交换器、消息队列和相关对象。虚拟主机是共享相同的身份认证和加密环境的独立服务器域。每个 vhost 本质上就是一个 mini 版的 RabbitMQ 服务器,拥有自己的队列、交换器、绑定和权限机制。vhost 是 AMQP 概念的基础,必须在连接时指定,RabbitMQ 默认的 vhost 是 / 。

    Broker:表示消息队列服务器实体

    二、消息中间件RabbitMQ---Docker安装RabbitMQ、以及RabbitMQ的基本使用【二】

    消息中间件RabbitMQ---Docker安装RabbitMQ、以及RabbitMQ的基本使用【二】

    1、安装

    1.1 拉取镜像

    docker pull rabbitmq:management

    1.2 查看镜像

    docker images

    1.3 创建容器

    1. docker run -d --name rabbitmq -p 5671:5671 -p 5672:5672 -p 4369:4369 -p
    2. 25672:25672 -p 15671:15671 -p 15672:15672 rabbitmq:management

    1.4 浏览器访问

    登录进去后的效果

    2、测试使用(基本步骤)

    2.1 创建交换机

    2.2 创建队列

    2.3 绑定

    3 、direct 交换器

    3.1 基本介绍

            消息中的路由键(routing key)如果和Binding 中的 binding key 一致, 交换器就将消息发到对应的队列中。路由键与队列名完全匹配,如果一个队列绑定到交换机要求路由键为“dog”,则只转发 routing key 标记为“dog”的消息,不会转发“dog.puppy”,也不会转发“dog.guard” 等等。它是完全匹配、单播的模式。

    3.2 测试

    3.2.1 创建direct类型交换机

    3.2.2 交换机和队列绑定

    3.2.3 发布消息

    3.2.4 队列查看


    4、fanout 交换器

    4.1 基本介绍

            每个发到 fanout 类型交换器的消息都会分到所有绑定的队列上去。fanout 交换器不处理路由键,只是简单的将队列绑定到交换器上,每个发送到交换器的消息都会被转发到与该交换器绑定的所有队列上。很像子网广播,每台子网内的主机都获得了一份复制的消息。fanout 类型转发消息是最快的。

    4.2 测试

    4.2.1 创建

    4.2.2 绑定

    4.2.3 发布消息

    4.2.4 队列查看


    5、topic 交换器

    5.1 基本介绍

             topic 交换器通过模式匹配分配消息的路由键属性,将路由键和某个模式进行匹配,此时队列需要绑定到一个模式上。它将路由键和绑定键的字符串切分成单词,这些单词之间用点隔开。它同样也会识别两个通配符:符号“#”和符号“”。#匹配0个或多个单词,匹配一个单词。

    5.2 测试

    5.2.1 创建

    5.2.2 绑定

    5.2.3 发布消息(都能收到)


    5.2.4 发布消息 (部分收到)


    三、消息中间件RabbitMQ---SpringBoot整合RabbitMQ【三】

    消息中间件RabbitMQ---SpringBoot整合RabbitMQ【三】

    1、SpringBoot整合

    1.1、pom依赖

    1. <dependency>
    2. <groupId>org.springframework.boot</groupId>
    3. <artifactId>spring-boot-starter-amqp</artifactId>
    4. </dependency>

    1.2、配置文件

    1. # RabbitMQ配置
    2. spring.rabbitmq.host=192.168.202.211
    3. spring.rabbitmq.port=5672
    4. # 虚拟主机配置
    5. spring.rabbitmq.virtual-host=/

    1.3、启动类开启

    1. @EnableRabbit
    2. @SpringBootApplication
    3. public class GulimallOrderApplication {
    4. public static void main(String[] args) {
    5. SpringApplication.run(GulimallOrderApplication.class, args);
    6. }
    7. }

    2、单元测试

    2.1 创建交换器

    1. /**
    2. * 1、如何创建Exchange、Queue、Binding
    3. * 1)、使用AmqpAdmin进行创建
    4. */
    5. @Test
    6. public void createExchange() {
    7. Exchange directExchange = new DirectExchange("hello-java-exchange",true,false);
    8. amqpAdmin.declareExchange(directExchange);
    9. log.info("Exchange[{}]创建成功:","hello-java-exchange");
    10. }

    使用代码创建和工具创建类似。


    创建结果

    2.2 创建队列

    1. @Test
    2. public void testCreateQueue() {
    3. Queue queue = new Queue("hello-java-queue",true,false,false);
    4. amqpAdmin.declareQueue(queue);
    5. log.info("Queue[{}]创建成功:","hello-java-queue");
    6. }

    创建结果

    2.3 创建绑定

    1. @Test
    2. public void createBinding() {
    3. Binding binding = new Binding("hello-java-queue",
    4. Binding.DestinationType.QUEUE,
    5. "hello-java-exchange",
    6. "hello.java",
    7. null);
    8. amqpAdmin.declareBinding(binding);
    9. log.info("Binding[{}]创建成功:","hello-java-binding");
    10. }

    结果

    2.4 发送消息1 (字符串)

    1. @Test
    2. public void sendMessageTestDemo1(){
    3. String msg = "hello java";
    4. rabbitTemplate.convertAndSend("hello-java-exchange","hello.java",msg);
    5. log.info("消息发送完成{}",msg);
    6. }

    查看接收的消息

    2.2 发送消息2 (对象)

    1. @Test
    2. public void sendMessageTest() {
    3. OrderReturnReasonEntity reasonEntity = new OrderReturnReasonEntity();
    4. reasonEntity.setId(1L);
    5. reasonEntity.setCreateTime(new Date());
    6. reasonEntity.setName("reason");
    7. reasonEntity.setStatus(1);
    8. reasonEntity.setSort(2);
    9. String msg = "Hello World";
    10. //1、发送消息,如果发送的消息是个对象,会使用序列化机制,将对象写出去,对象必须实现Serializable接口
    11. //2、发送的对象类型的消息,可以是一个json
    12. rabbitTemplate.convertAndSend("hello-java-exchange","hello.java",
    13. reasonEntity,new CorrelationData(UUID.randomUUID().toString()));
    14. log.info("消息发送完成:{}",reasonEntity);
    15. }


    由于是序列化,需要添加配置,经过转换后才可以看到对象数据

    1. @Configuration
    2. public class MyRabbitConfig {
    3. @Bean
    4. public MessageConverter messageConverter() {
    5. return new Jackson2JsonMessageConverter();
    6. }
    7. }

    2.3 接收数据

    1. /**
    2. * queues:声明需要监听的队列
    3. * channel:当前传输数据的通道
    4. */
    5. @RabbitListener(queues = {"hello-java-queue"})
    6. public void revieveMessage(Message message,
    7. OrderReturnReasonEntity content) {
    8. //拿到主体内容
    9. byte[] body = message.getBody();
    10. //拿到的消息头属性信息
    11. MessageProperties messageProperties = message.getMessageProperties();
    12. System.out.println("接受到的消息...内容" + message + "===内容:" + content);
    13. }

  • 相关阅读:
    红蓝对抗-攻防演练中红队如何识别蜜罐保护自己
    crane:字典项与关联数据处理的新思路
    [极客大挑战 2019]BabySQL 1
    LLM 大模型学习必知必会系列(十二):VLLM性能飞跃部署实践:从推理加速到高效部署的全方位优化[更多内容:XInference/FastChat等框架]
    Scaling Up Your Kernels to 31x31: Revisiting Large Kernel Design in CNNs笔记
    生命在于学习——主板跳线的学习
    protobuf在linux下载编译和使用
    np.argsort()函数
    为了面试大厂,熬夜肝完这份 “测试” 笔记后,我终于“硬”了一回
    扬帆际海—shopee跨境店和本土店谁更有优势?
  • 原文地址:https://blog.csdn.net/weixin_43304253/article/details/133621425