RabbitMq消息可靠性之回退模式 通俗易懂 超详细 【内含案例】

RabbitMq保证消息可靠性之回退模式

介绍

生产者生产的消息没有正确的到达队列就会触发回退模式,进行二次发送

前提

完成 SpringBoot 整合 RabbitMq 中的Topic通配符模式

一、更改Producer工程的application.yml文件

spring:
  rabbitmq:
    host: localhost
    port: 5672
    virtual-host: /
    username: username
    password: password
    publisher-returns: true #开启回退模式
server:
  port: 8080

二、更改ProducerTest.java文件 ConfirmCallback

import lombok.extern.slf4j.Slf4j;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.context.annotation.Scope;
import org.springframework.test.context.junit4.SpringRunner;

import javax.annotation.Resource;


@RunWith(SpringRunner.class)
@SpringBootTest
@Slf4j
public class RabbitMqTest {
    
    private RabbitTemplate rabbitTemplate;

    @Resource
    @Scope("prototype")
    public void setRabbitTemplate(RabbitTemplate rabbitTemplate) {
        /*
         SpringBoot 默认为true 如果为false下面不执行
         rabbitTemplate.setMandatory(true);
         注意:Spring项目必须书写!
         */

        /**
         * 路由没有到达Queue就会执行
         */
        rabbitTemplate.setReturnCallback(new RabbitTemplate.ReturnCallback() {
            /**
             *
             * @param message 消息对象
             * @param replyCode 消息编码
             * @param replyText 时报错误信息
             * @param exchange 交换机
             * @param routingKey 路由键
             */
            @Override
            public void returnedMessage(Message message, int replyCode, String replyText, String exchange, String routingKey) {
                log.debug("消息没有到达Queue,该消息为:{}",message.getBody());
                log.debug("错误编码:{},错误信息:{}",replyCode,replyText);
                log.debug("交换机:{},路由键:{}",exchange,routingKey);
                //这里routing填写正确的
                rabbitTemplate.send("topic_exchange","item.aa",message);
            }
        });

        this.rabbitTemplate = rabbitTemplate;
    }

    @Test
    public void test() throws InterruptedException {
        String body = "回退模式发送消息";
        //为了达到回退模式 ,routingKey 填写一个错误的 会调用 ReturnCallback 发送一个正确的
        rabbitTemplate.convertAndSend("topic_exchange","dsafasf56.chu",body);
        Thread.sleep(2000);
    }
}

三、测试

首先运行 ProducerTest.java 单元测试,然后在启动 ConsumerListener.java 消息监听器

  1. 如果已经存在 topic_queue 请先删除后再执行单元测试

四、小结

第一次发送的消息不会到达queue,会调用到 ReturnCallback 方法,会再次进行发送.保证消息的可靠性,不会丢失.

RabbitMQ消息可靠性是指确保消息可以安全、可靠地传递到消费者。为了实现消息可靠性RabbitMQ提供了以下机制: 1. 持久化队列:通过将队列设置为持久化,即使RabbitMQ服务器重启,队列中的消息也不会丢失。 2. 持久化消息:将消息设置为持久化,使得即使在RabbitMQ服务器重启前,消息也会被存储在磁盘上。 3. 消息确认机制:生产者可以通过消息确认机制来确保消息已经被成功发送到RabbitMQ中。当消息成功地被RabbitMQ接收到后,生产者会收到一个确认信号。如果RabbitMQ在处理消息时发生错误,生产者可以根据确认信号来重新发送消息。 尽管RabbitMQ提供了上述机制,但仍然存在一些情况下消息可能丢失的风险。例如,如果消息RabbitMQ服务器接收到但尚未持久化到磁盘上时,RabbitMQ服务器崩溃,这可能导致部分消息的丢失。 为了进一步提高消息可靠性,可以采取以下措施: 1. 使用事务:使用事务可以确保消息的原子性提交,即要么全部成功发送,要么全部失败回滚。但是,使用事务会降低RabbitMQ的性能。 2. 设置消息确认模式:可以将消息确认模式设置为"confirm",使得RabbitMQ在收到消息后立即发送确认信号给生产者。 3. 设置备份队列:备份队列可以在主队列发生故障时,将消息转发到备份队列,从而避免消息的丢失。 4. 通过持久化到数据库或其他存储系统来保存重要的消息数据。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值