RabbitMQ

RabbitMQ

1. 安装教程

  1. windows平台

Erlang安装:Erlang安装_erlang安装教程-CSDN博客

Erlang和RabbitMQ版本兼容:windows安装rabbitmq和环境erlang(最详细版,包括对应关系,安装错误解决方法)_erlang和rabbitmq关系-CSDN博客

  1. linux

docker平台安装rabbitmq:

Linux平台使用Docker安装RabbitMQ - ooyhao - 博客园 (cnblogs.com)

Docker中安装RabbitMQ - niceyoo - 博客园 (cnblogs.com)

Docker下RabbitMQ安装 - work hard work smart - 博客园 (cnblogs.com)

docker run -d --hostname my-rabbitmq \
-p 15672:15672 -p 5672:5672 \
--name my-rabbitmq \
-e RABBITMQ_DEFAULT_USER=admin \
-e RABBITMQ_DEFAULT_PASS=admin \
rabbitmq:3.8.9-management

关于参数:

  • -d:以守护进程方式在后台运行
  • --hostname:设置容器的主机名称,仅本机可见。这种方式是写到 /etc/hostname ,以及 /etc/hosts 文件中,作为容器主机IP的别名,并且将显示在容器的bash中。不过这种方式设置的主机名将不容易被容器之外可见。这将不会出现在 docker ps 或者 其他的容器的 /etc/hosts 文件中。
  • --name:指定容器名
  • -p:指定服务运行的端口。
  • RABBITMQ_DEFAULT_USER:用于设置登陆控制台的用户名,这里我设置 admin
  • RABBITMQ_DEFAULT_PASS:用于设置登陆控制台的密码,这里我设置 admin容器启动成功后,可以在浏览器输入地址:http://ip:15672/访问控制台

额外说一下 rabbitmq 这几个端口的作用:

  • 4369:EPMD( Erlang Port Mapper Daemon)端口号,在 Erlang 集群中相当于 dns 的作用
  • 5672:client 端通信端口
  • 15672:web管理界面端口

2. 详细笔记

  1. 黑马详细笔记:入门

黑马程序员-RabbitMQ | WeiBlog (gitee.io)

RabbitMQ (yuque.com)

  1. 其他详细笔记

命令行与管控台 (yuque.com)

新RabbitMQ精讲 提升工程实践能力 培养架构思维 (yuque.com)

手把手教你SpringBoot集成消息服务中间件RabbitMQ - 你樊不樊 - 博客园 (cnblogs.com)

3. rabbitmq 6种队列模式

RabbitMQ - 随笔分类 - niceyoo - 博客园 (cnblogs.com)

黑马程序员-RabbitMQ | WeiBlog (gitee.io)

rabbitmq详细实例 - JiuYou2020 - 博客园 (cnblogs.com)

4. Rabbitmq模块实现

来源:个人做了一个Spring Boot整合RabbitMQ的笔记
含工作、发布订阅、路由、主题模式示例
RabbitMQ消息可靠投递,可靠消费。完全可以在工作中直接cv使用。
github地址:https://github.com/WENZIZZHENG/spring-boot-demo
gitee地址:https://gitee.com/BiMon/spring-boot-demo

5. 学习遇到问题

使用Idea创建一个父子SpringBoot项目 - 飘杨...... - 博客园 (cnblogs.com)

6. springboot 集成rabbitmq

1. 生产者

  1. 导入坐标

详解Spring Boot 项目中的 parent_java_脚本之家 (jb51.net)

SpringBoot到底是什么?如何理解parent、starter、引导类以及内嵌Tomcat? - 知乎 (zhihu.com)

  <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.6.13</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
<dependency>
        <groupId>org.springframework.boot</groupId>
         <artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
  1. rabbitmq配置
spring:
  rabbitmq:
    host: 8.*.*.213
    username: admin
    password: admin
    virtual-host: /rabbitmq_springboot
    port: 5672
  1. rabbitmqConfig编写
@Configuration
public class RabbitConfig {

    public static final String EXCHANGE_NAME="boot_topic_exchange";
    public static final String QUEUE_NAME="boot_queue";


    //交换机
    @Bean("bootExchange")
    public Exchange bootExchange(){
        return ExchangeBuilder.topicExchange(EXCHANGE_NAME).durable(true).build();
    }

    //2.Queue队列
    @Bean("bootQueue")
    public Queue bootQueue(){
        return QueueBuilder.durable(QUEUE_NAME).build();
    }

    //3.队列和交换就绑定关系 Binding
    /*
    1.知道哪个队列
    2.知道哪个交换机
    3.routing key
     */

    @Bean
    public Binding bindingQueueExchange(@Qualifier("bootQueue")Queue queue,@Qualifier("bootExchange")Exchange exchange){
        return BindingBuilder.bind(queue).to(exchange).with("boot.#").noargs();
    }
}

  1. 测试
@SpringBootTest
class BootproducerApplicationTests {
    @Resource
    private RabbitTemplate rabbitTemplate;//注入rabbitmq模板类
    @Test
    void springbootRabbitMq(){
        rabbitTemplate.convertAndSend(RabbitConfig.EXCHANGE_NAME,"boot.hahah","boot mq hello....");
    }
}

2. 消费者

  1. 导入依赖
  <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.6.13</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
<dependency>
        <groupId>org.springframework.boot</groupId>
         <artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
spring:
  rabbitmq:
    host: 8.*.*.213
    username: admin
    password: admin
    virtual-host: /rabbitmq_springboot
    port: 5672
  1. 编写消费者

@Component
public class RabbitMQListener {
    @RabbitListener(queues = "boot_queue")
    public void ListenerQueue(Message message) {
        System.out.println(message);
        System.out.println(message.getBody());
    }
}

7.rabbitmq 高级篇

1. 生产者消息可靠投递:confirm机制

  1. confirm 确认流程

RabbitMQ消息可靠性传输 - 一步一年 - 博客园 (cnblogs.com)

image-20240119164057964

  1. 代码编写

1. 生产者配置yml 文件

spring:
  rabbitmq:
       publisher-returns: true # return 确认机制
    template:
      mandatory: true
    publisher-confirm-type: simple

publisher-returns: 开启消息从exhcange路由到queue的回调,只有路由失败时才会触发回调

mandatory: 为true时,如果exchange根据routingKey将消息路由到queue时找不到匹配的queue,触发return回调,为false时,exchange直接丢弃消息。

publisher-confirm-type: 表示确认消息的类型,分别有none、correlated、simple这三种类型。

  1. none:表示禁用发布确认模式

  2. correlated:表示消息成功到达Broker后触发ConfirmCalllBack回调

  3. simple: simple模式下如果消息成功到达Broker后一样会触发ConfirmCalllBack回调,发布消息成功后使用rabbitTemplate调用waitForConfirms或waitForConfirmsOrDie方法等待broker节点返回发送结果

RabbitMQ相关--消息确定机制_publisher-confirm-type-CSDN博客

2. 编写配置类

/* description
 * 1.RabbitTemplate.ConfirmCallback 消息被正常发送到交换机, 则会调用该方法, 自动回调
 * 2.RabbitTemplate.ReturnsCallback 消息被交换机正常发送到队列,则会回调改方法, 自动回调
 */
@Component
@Slf4j
public class RabbitMQConfirmAndReturnConfig implements RabbitTemplate.ConfirmCallback,RabbitTemplate.ReturnsCallback {
    @Autowired
    private RabbitTemplate rabbitTemplate;
    @PostConstruct
    public void initMethod(){
        rabbitTemplate.setConfirmCallback(this);
        rabbitTemplate.setReturnsCallback(this);
    }
    /**
     * @description:
     *
     * @param correlationData: 相关数据,对象中有一个id的属性,用来表示消息的唯一标识
     * @param ack: true:表示正常发送成功
     * @param cause 表示消息发送失败原因
     * @author life
     * @date 2024-01-19 16:55
     * @return void
     */
    @Override
    public void confirm(CorrelationData correlationData, boolean ack, String cause) {
        if(ack){
            System.out.println("生产者到交换机成功");
        }else{
            System.out.println("消息发送到交换机失败");
        }

    }
    //RabbitTemplate.ReturnsCallback
    /**
     * @description:
     *消息发功成功,则会不调用改方法, 消息失败 则调用该方法
     * @param returnedMessage:
     *  -message: 发送消息内容
     *  - replyCode: 回应码
     *  - replyText: 回应内容
     *  - exchange:交换机
     *  - routingKey: 路由键
     * @author life
     * @date 2024-01-19 16:58
     * @return void
     */
    @Override
    public void returnedMessage(ReturnedMessage returnedMessage) {
        System.out.println("交换机到队列失败");
    }
}

参考博客

视频1.RabbitMQ消息确认机制介绍_哔哩哔哩_bilibili

RabbitMQ消息可靠性传输 - 一步一年 - 博客园 (cnblogs.com)

RabbitMQ相关--消息确定机制_publisher-confirm-type-CSDN博客

2. 消费者如何保证消息不丢失:手动提交ack

  1. 存在情况

消费者接收到消息,但是还未处理或者还未处理完,此时消费者进程挂掉了,比如重启或者异常断电等,此时mq认为消费者已经完成消息消费,就会从队列中删除消息,从而导致消息丢失。

  1. 解决办法

RabbitMQ默认是自动ack的,此时需要将其修改为手动ack,也即自己的程序确定消息已经处理完成后,手动提交ack,此时如果再遇到消息未处理进程就挂掉的情况,由于没有提交ack,RabbitMQ就不会删除这条消息,而是会把这条消息发送给其他消费者处理,但是消息是不会丢的。

  1. 具体实现办法

消费者application.yaml:配置信息

spring:
  rabbitmq:
    listener:
      simple:
        acknowledge-mode: manual  # 手动ack
        prefetch: 5 # 消峰 最多有5个

参数

image-20240120132603214

消费者手动确认

@Component
public class RabbitMQListener {

    @RabbitListener(queues = "boot_queue")
    public void ListenerQueue(Message message, Channel channel) throws IOException {
        try{
            System.out.println(message);
            int a=1/0;
            // 确认回复收到 true:批量确认消息  false 单独确认消息
            channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);
        }catch (Exception e){
            // 参数2  true 批量确认消息 false  单独确认消息
            // 参数3 true 重发 插入末尾  false 队列丢弃
            channel.basicNack(message.getMessageProperties().getDeliveryTag(),false,true);
        }
        
    }
}

对于channel.basicNack方法的第三个参数,表示消息nack后是否返回队列,如果设置为true,表示返回队列,此时消息处于队列头部,消费者会一直处理该消息,影响后续消息的消费,设置为false时表示不返回队列,此时如果设置有DLX(死信队列),那么消息会进入DLX中,后续再对该消息进行相应的处理,如果没有设置DLX,此时消息就会被丢弃。

3. 消费端限流

  1. 演示

image-20240120134549000

image-20240120134600714

  1. 实现

applicaton.yaml配置

spring:
  rabbitmq:
    listener:
      simple:
        acknowledge-mode: manual # 手动确认
        prefetch: 5 # 消减峰值
  1. 注意事项

image-20240120134726129

4. 死信队列

  1. 死信队列

死信,Dead Letter,一种消息机制,当消费者去消费队列中的消息时,如果队列中的消息出现了以下的情况:

  • 消费端执行nack或者reject时,设置requeue=false;: 消费者拒绝接受
  • 消息在队列中的时间超过设置的TTL(Time To Live)时间;:消息过期
  • 队列中消息的数量超过设置的最大数量;: 队列满,再添加消息

那么这些消息就可以被称之为死信消息,在配置了死信队列的情况下,死信消息会进入死信队列,如果没有配置死信队列,这些死信消息会被丢弃

死信队列和普通队列

死信交换机可以是fanout、direct、topic等类型,和普通交换机并无不同;

死信交换机要绑定要业务队列上才会生效;

给死信交换机绑定的队列称之为死信队列,其实就是普通的队列,没有任何特殊之处

  1. 死信队列实现

架构图

image-20240121132814693

生产者

@Configuration
public class RabbitDLConfig {
    public static final String EXCHANGE_NAME="boot_topic_exchange";//普通交换机
    public static final String DL_EXCHANGE_NAME="topic_exchange_deadLetter";//死信交换机
    public static final String QUEUE_NAME="boot_queue";//普通队列
    public static final String DL_QUEUE_NAME="queue_deadLetter";//死信交换机
	//普通队列
    //1. 交换机
    @Bean("bootExchange")
    public Exchange bootExchange(){
        return ExchangeBuilder.topicExchange(EXCHANGE_NAME).durable(true).build();
    }

    //2.Queue队列
    @Bean("bootQueue")
    public Queue bootQueue(){
        return QueueBuilder
                .durable(QUEUE_NAME)
                .deadLetterExchange(DL_EXCHANGE_NAME)//绑定死信交换机
                .deadLetterRoutingKey("dl.orginal")// 指定路由键
                .build();
    }

    //3.队列和交换就绑定关系 Binding
    /*
    1.知道哪个队列
    2.知道哪个交换机
    3.routing key
     */

    @Bean
    public Binding bindingQueueExchange(@Qualifier("bootQueue")Queue queue, @Qualifier("bootExchange")Exchange exchange){
        return BindingBuilder.bind(queue).to(exchange).with("boot.#").noargs();
    }



    // 死信队列
    //1. 死信交换机
    @Bean("deadLetterExchange")
    public Exchange deadLetterExchange(){
        return ExchangeBuilder.topicExchange(DL_EXCHANGE_NAME).durable(true).build();
    }

    //2.死信Queue队列
    @Bean("deadLetterQueue")
    public Queue deadLetterQueue(){
        return QueueBuilder.durable(DL_QUEUE_NAME).build();
    }

    //3.队列和交换就绑定关系 Binding
    /*
    1.知道哪个队列
    2.知道哪个交换机
    3.routing key
     */
    //死信队列绑定死信交换机
    @Bean
    public Binding bindingDLQueueExchange(@Qualifier("deadLetterQueue")Queue queue, @Qualifier("deadLetterExchange")Exchange exchange) {
        return BindingBuilder.bind(queue).to(exchange).with("dl.#").noargs();
    }


}

普通队列消费者

@Component
public class RabbitMQListener {

    @RabbitListener(queues = "boot_queue")
    public void ListenerQueue(Message message, Channel channel) throws IOException {
        System.out.println("我是普通消费者:"+message.getBody());
    }
}

死信队列消费者

// 死信队列
@Component
public class DeadLetterRQListener {
    @RabbitListener(queues={"queue_deadLetter"})// 绑定死信队列
    public void ListenerDeadQueue(Message message, Channel channel){
        System.out.println("我是死信队列消费者:"+message.getBody());
    }
}

生产者测试 消息过期:

@SpringBootTest
public class DeadLetterQueueTest {
    @Resource
    private RabbitTemplate rabbitTemplate;

    @Test
    public void testDeadQueue(){
        MessagePostProcessor messagePostProcessor = message ->  {
                //1.设置message的信息
                message.getMessageProperties().setExpiration("1");//消息的过期时间
                //2.返回该消息
                return message;
        };
        for(int i=0;i<10;i++){
            String time = LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyyMMdd"));
            String msg="生产者发送第"+i+"条消息给消费者"+"时间:"+time;
            if(i%2==0){     rabbitTemplate.convertAndSend(RabbitDLConfig.EXCHANGE_NAME,"boot.hahah",msg);
            }else{   rabbitTemplate.convertAndSend(RabbitDLConfig.EXCHANGE_NAME,"boot.hahah",msg,  messagePostProcessor);
            }
        }
    }
}

image-20240120153249631

私信队列总结

死信队列更多的是用来保证消息的可靠性,主要用于比较重要的队列,用以确保未被正确消费的消息不会丢失,其实也可以不用死信队列,在消费端出现异常时,可以将消息从当前队列ack掉,再将其发送到其他队列,然后再单独处理其他队列

RabbitMQ死信队列 - 一步一年 - 博客园 (cnblogs.com)

5.延迟队列

  1. 什么是延迟队列

延时队列中的元素是希望在指定时间到了以后或之前取出和处理,简单来说,延时队列就是用来存放需要在指定时间被处理的元素的队列。

  1. 使用场景
  1. 订单在十分钟之内未支付则自动取消
  2. 新创建的店铺,如果在十天内都没有上传过商品,则自动发送消息提醒。
  3. 用户注册成功后,如果三天内没有登陆则进行短信提醒。
  4. 用户发起退款,如果三天内没有得到处理则通知相关运营人员。
  5. 预定会议后,需要在预定的时间点前十分钟通知各个与会人员参加会议

如何将未付款的订单取消?

方法 一 :定时任务轮询查找数据。数据量小的时候可以

方法二: 消息队列延时队列解决。

  1. 延时队列实现一:死信队列+ttl 过期时间

死信队列+ttl 过期时间

image-20240121133322750

  1. 延时队列实现二:延时队列插件

1. 下载插件

docker安装RabbitMQ及安装延迟插件的详细过程_docker_脚本之家 (jb51.net)

#刚刚上传的插件拷贝到容器内plugins目录下
# docker cp 当前插件地址 容器名字:/plugins
docker cp /root/rabbitmq_delayed_message_exchange-3.8.9-0199d11c.ez my-rabbitmq:/plugins
# 进入容器
docker exec -it my-rabbitmq /bin/bash

2. 启用插件,重启容器

# 启动延迟插件
rabbitmq-plugins enable rabbitmq_delayed_message_exchange

image-20240121135330302


[root@iZ2vc2rht08hck1x6us154Z ~]# docker restart my-rabbitmq

3. 查看是否安装成功

image-20240121135621195

4.交换机配置修改

image-20240121133944700

5. 生产者 setExpiration 改成setDeLay

image-20240121140246313

博客链接

八、RabbitMq死信队列与延迟队列 - hp柠檬茶 - 博客园 (cnblogs.com)

MQ高级-11.延迟消息-死信交换机_哔哩哔哩_bilibili

6.幂等性:解决重复消费

  1. 什么是幂等性

用户对于同一操作发起的一次请求或者多次请求的结果是一致的。

例子:退款业务出现问题,多次退款 恢复多次余额

image-20240121124830553

  1. 消息重复消费

消费者在消费 MQ 中的消息时,MQ 已把消息发送给消费者消费者在给 MQ 返回 ack 时网络中断故 MQ 未收到确认信息,该条消息会重新发给其他的消费者,或者在网络重连后再次发送给该消费者但实际上该消费者已成功消费了该条消息,造成消费者消费了重复的消息

  1. 幂等性解决办法

1.唯一消息id

image-20240121125243361

image-20240121125401654

image-20240121125658400

  1. MQ 消费者的幂等性的解决一般使用全局 ID 或者写个唯一标识比如时间戳 或者 UUID 或者订单消费者消费 MQ 中的消息也可利用 MQ 的该 id 来判断,
  2. 或者可按自己的规则生成一个全局唯一 id,每次消费消息时用该 id 先判断该消息是否已消费过
  1. 消费端的幂等性保障

a. 唯一 ID+指纹码机制,利用数据库主键去重

b. 利用 redis 的原子性去实现:利用 redis 执行 setnx 命令,天然具有幂等性。从而实现不重复消费

博客链接

十、RabbitMQ 幂等性 - hp柠檬茶 - 博客园 (cnblogs.com)

7.docker搭建rabbitmq 集群

  1. 拉取rabbitmq镜像
docker pull rabbitmq:3.8.9-management
  1. 启动多个rabbitmq容器

主机1

docker run -d --hostname my-rabbitmq \
-p 15672:15672 -p 5672:5672 \
--name my-rabbitmq \
-e RABBITMQ_DEFAULT_USER=admin \
-e RABBITMQ_DEFAULT_PASS=admin \
-e RABBITMQ_ERLANG_COOKIE='admin'\
 rabbitmq:3.8.9-management
  • -e RABBITMQ_ERLANG_COOKIE='rabbit_cluster' 设置rabbitmq的cookie,该值可以任意设置,只需要三个容器保持一致即可 :集群之间相互通信 cookie一致
  • --link 容器名:主机名 作用不同主机之间相互通信

主机2

docker run -d --hostname my-rabbitmq1 \
-p 15673:15672 -p 5673:5672 \
--name my-rabbitmq1 \
--link my-rabbitmq:my-rabbitmq \
-e RABBITMQ_DEFAULT_USER=admin \
-e RABBITMQ_DEFAULT_PASS=admin \
-e RABBITMQ_ERLANG_COOKIE='admin'\
 rabbitmq:3.8.9-management

主机2和主机1 连接

主机3

docker run -d --hostname my-rabbitmq2 \
-p 15674:15672 -p 5674:5672 \
--name my-rabbitmq2 \
--link my-rabbitmq:my-rabbitmq \
--link my-rabbitmq1:my-rabbitmq1 \
-e RABBITMQ_DEFAULT_USER=admin \
-e RABBITMQ_DEFAULT_PASS=admin \
-e RABBITMQ_ERLANG_COOKIE='admin'\
 rabbitmq:3.8.9-management

主机3和主机2 和主机1链接

image-20240120155737432

  1. 配置集群

重置节点一

docker exec -it my-rabbitmq /bin/bash 
rabbitmqctl stop_app && rabbitmqctl reset && rabbitmqctl start_app 
exit

image-20240120161551340

节点一名称:rabbit@my-rabbitmq

重置节点二

docker exec -it my-rabbitmq1 /bin/bash 
rabbitmqctl stop_app && rabbitmqctl reset
rabbitmqctl join_cluster rabbit@my-rabbitmq
rabbitmqctl start_app
exit

image-20240120161914644

节点一名称:rabbit@my-rabbitmq1

重置节点三

docker exec -it my-rabbitmq2 /bin/bash 
rabbitmqctl stop_app && rabbitmqctl reset
rabbitmqctl join_cluster rabbit@my-rabbitmq1
rabbitmqctl start_app
exit

image-20240120162247981

  1. 查看集群状态
rabbitmqctl cluster_status

image-20240120162350920

  1. 浏览器查看集群状态

image-20240120162720061

  1. 关闭其他节点

image-20240120163416932

多台服务器配置rabbit集群: Docker环境RabbitMq配置SSL - 一步一年 - 博客园 (cnblogs.com)

21rabbit集群搭建_哔哩哔哩_bilibili

8. rabbitmq

订单服务完善不可路由消息处理 (yuque.com)

  1. 封装消息
/* 发送消息 */

        //1. 设置消息的编码和内容格式
        MessageProperties properties = new MessageProperties();
        properties.setContentType(MessageProperties.CONTENT_TYPE_JSON);
        properties.setContentEncoding("utf-8");
        //2. 消息对象转成json字符串
        String jsonMsg = JSON.toJSONString(messageDTO);
        //3. 将所需要发送的消息 ,以及消息的格式封装成Message对象
        Message sendMsg = new Message(jsonMsg.getBytes(StandardCharsets.UTF_8),properties);
        //4. Correlateion 设置returnMessage失败返回内容
        CorrelationData correlationData = new CorrelationData();
        ReturnedMessage returnedMessage = new ReturnedMessage(sendMsg,300,"餐厅收到订单消息,发送给订单服务队列",
                RabbitConfig.RESTAURANT_EXCHANGE,
                "key.order");
        correlationData.setReturned(returnedMessage);
        correlationData.setId(messageDTO.getOrderID());     //设置为订单ID
        //5. 生产者发送消息 
        rabbitTemplate.convertAndSend(RabbitConfig.RESTAURANT_EXCHANGE,"key.order",sendMsg,correlationData);

        // 6. 消费者手动确认消息答
        channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);        //手动消息应答

面试题

rabbitMQ 消息顺序性、消息幂等性、消息不丢失、最终一致性、补偿机制、消息队列设计_消息补偿机制-CSDN博客

相关参考

视频参考

rabbitmq 视频21rabbit集群搭建_哔哩哔哩_bilibili

面试题RabbitMQ面试题详解 | 一口气看完16个面试必问的rabbitmq面试核心知识点,让你面试少走99%的弯路!_哔哩哔哩_bilibili

博客推荐

RabbitMQ消息可靠性传输 - 一步一年 - 博客园 (cnblogs.com)

posted @ 2024-01-25 20:38  我爱读论文  阅读(35)  评论(0编辑  收藏  举报