SpringBoot+RabbitMQ发送确认和消费手动确认机制

本文详细介绍RabbitMQ中消息的发送确认与消费手动确认机制。通过配置Spring Boot的RabbitMQ属性,实现消息发送至Exchange及Queue的确认,以及消费端的手动确认流程,确保消息传递的可靠性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

  • 配置RabbitMQ
    # 发送确认
    spring.rabbitmq.publisher-confirms=true
    # 发送回调
    spring.rabbitmq.publisher-returns=true
    # 消费手动确认
    spring.rabbitmq.listener.simple.acknowledge-mode=manual

    1. 生产发送消息确认机制

  • 其实这个也不能叫确认机制,只是起到一个监听的作用,监听生产者是否发送消息到exchange和queue。
  • 生产者和消费者代码不改变。
  • 新建配置类 MQProducerAckConfig.java 实现ConfirmCallback和ReturnCallback接口,@Component注册成组件。
  • ConfirmCallback只确认消息是否到达exchange,已实现方法confirm中ack属性为标准,true到达,反之进入黑洞。
  • ReturnCallback消息没有正确到达队列时触发回调,如果正确到达队列不执行。
  • package com.fzb.rabbitmq.config;
    
    import org.apache.commons.lang3.SerializationUtils;
    import org.springframework.amqp.core.Message;
    import org.springframework.amqp.rabbit.connection.CorrelationData;
    import org.springframework.amqp.rabbit.core.RabbitTemplate;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Component;
    
    import javax.annotation.PostConstruct;
    
    /**
     * @Description 消息发送确认
     * <p>
     * ConfirmCallback  只确认消息是否正确到达 Exchange 中
     * ReturnCallback   消息没有正确到达队列时触发回调,如果正确到达队列不执行
     * <p>
     * 1. 如果消息没有到exchange,则confirm回调,ack=false
     * 2. 如果消息到达exchange,则confirm回调,ack=true
     * 3. exchange到queue成功,则不回调return
     * 4. exchange到queue失败,则回调return
     * @Author jxb
     * @Date 2019-04-04 16:57:04
     */
    @Component
    public class MQProducerAckConfig implements RabbitTemplate.ConfirmCallback, RabbitTemplate.ReturnCallback {
    
        @Autowired
        private RabbitTemplate rabbitTemplate;
    
        @PostConstruct
        public void init() {
            rabbitTemplate.setConfirmCallback(this);            //指定 ConfirmCallback
            rabbitTemplate.setReturnCallback(this);             //指定 ReturnCallback
    
        }
    
        @Override
        public void confirm(CorrelationData correlationData, boolean ack, String cause) {
            if (ack) {
                System.out.println("消息发送成功" + correlationData);
            } else {
                System.out.println("消息发送失败:" + cause);
            }
        }
    
        @Override
        public void returnedMessage(Message message, int replyCode, String replyText, String exchange, String routingKey) {
            // 反序列化对象输出
            System.out.println("消息主体: " + SerializationUtils.deserialize(message.getBody()));
            System.out.println("应答码: " + replyCode);
            System.out.println("描述:" + replyText);
            System.out.println("消息使用的交换器 exchange : " + exchange);
            System.out.println("消息使用的路由键 routing : " + routingKey);
        }
    }

    2. 消费消息手动确认

  • SpringBoot集成RabbitMQ确认机制分为三种:none、auto(默认)、manual
    Auto
    1. 如果消息成功被消费(成功的意思是在消费的过程中没有抛出异常),则自动确认
    2. 当抛出 AmqpRejectAndDontRequeueException 异常的时候,则消息会被拒绝,且 requeue = false(不重新入队列)
    3. 当抛出 ImmediateAcknowledgeAmqpException 异常,则消费者会被确认
    4. 其他的异常,则消息会被拒绝,且 requeue = true,此时会发生死循环,可以通过 setDefaultRequeueRejected(默认是true)去设置抛弃消息

     

  • 如设置成manual手动确认,一定要对消息做出应答,否则rabbit认为当前队列没有消费完成,将不再继续向该队列发送消息。
  • channel.basicAck(long,boolean); 确认收到消息,消息将被队列移除,false只确认当前consumer一个消息收到,true确认所有consumer获得的消息。
  • channel.basicNack(long,boolean,boolean); 确认否定消息,第一个boolean表示一个consumer还是所有,第二个boolean表示requeue是否重新回到队列,true重新入队。channel.basicReject(long,boolean); 拒绝消息,requeue=false 表示不再重新入队,如果配置了死信队列则进入死信队列。
  • 当消息回滚到消息队列时,这条消息不会回到队列尾部,而是仍是在队列头部,这时消费者会又接收到这条消息,如果想消息进入队尾,须确认消息后再次发送消息。

尊重原创,原文地址:https://www.jianshu.com/p/fae8fca98522

 

评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值