IT评测·应用市场-qidao123.com

标题: RabbitMQ如何包管消息可靠 [打印本页]

作者: tsx81428    时间: 2024-7-20 18:32
标题: RabbitMQ如何包管消息可靠
解决办法: 
1、做好消息确认机制(pulisher、consumer[手动ACK]
2、每一个发送的消息都在数据库做好记录。定期将失败的消息再次发送一遍 
消息确认机制: 
生产者确认模式:确认消息是否发送到broker,失败缘故原由是什么。配置类@PostConstruct方法里,调用setConfirmCallback()方法,参数是Lambda表达式
生产者退回模式:确认消息是否发送到队列。配置类@PostConstruct方法里,调用setReturnCallback()方法,参数是Lambda表达式
消耗者ack机制:消耗者方法的Channel参数、Message参数、消息实体类参数。肯定要手动ack,消耗乐成才移除消息。
  1. /**
  2.      * 定制RabbitTemplate
  3.      * 1、服务器收到消息就回调
  4.      *      1、spring.rabbitmq.publisher-confirms=true
  5.      *      2、设置确认回调ConfirmCallback
  6.      * 2、消息正确抵达队列进行回调
  7.      *      1、 spring.rabbitmq.publisher-returns=true
  8.      *          spring.rabbitmq.template.mandatory=true
  9.      *      2、设置确认回调ReturnCallback
  10.      *
  11.      * 3、消费端确认(保证每个消息被正确消费,此时才可以broker删除这个消息)。
  12.      *      spring.rabbitmq.listener.simple.acknowledge-mode=manual 手动签收
  13.      *      1、默认是自动确认的,只要消息接收到,客户端会自动确认,服务端就会移除这个消息
  14.      *          问题:
  15.      *              我们收到很多消息,自动回复给服务器ack,只有一个消息处理成功,宕机了。就会发生消息丢失;
  16.      *              消费者手动确认模式。只要我们没有明确告诉MQ,货物被签收。没有Ack,
  17.      *                  消息就一直是unacked状态。即使Consumer宕机。消息不会丢失,会重新变为Ready,下一次有新的Consumer连接进来就发给他
  18.      *      2、如何签收:
  19.      *          channel.basicAck(deliveryTag,false);签收;业务成功完成就应该签收
  20.      *          channel.basicNack(deliveryTag,false,true);拒签;业务失败,拒签
  21.      */
  22. //    @PostConstruct //MyRabbitConfig对象创建完成以后,执行这个方法
  23.     public void initRabbitTemplate(){
  24.         //设置确认回调
  25.         rabbitTemplate.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {
  26.             /**
  27.              *
  28.              * 1、只要消息抵达Broker就ack=true
  29.              * @param correlationData 当前消息的唯一关联数据(这个是消息的唯一id)
  30.              * @param ack  消息是否成功收到
  31.              * @param cause 失败的原因
  32.              */
  33.             @Override
  34.             public void confirm(CorrelationData correlationData, boolean ack, String cause) {
  35.                 /**
  36.                  * 1、做好消息确认机制(pulisher,consumer【手动ack】)
  37.                  * 2、每一个发送的消息都在数据库做好记录。定期将失败的消息再次发送一遍
  38.                  */
  39.                 //服务器收到了;
  40.                 //修改消息的状态
  41.                 System.out.println("confirm...correlationData["+correlationData+"]==>ack["+ack+"]==>cause["+cause+"]");
  42.             }
  43.         });
  44.         //设置消息抵达队列的确认回调
  45.         rabbitTemplate.setReturnCallback(new RabbitTemplate.ReturnCallback() {
  46.             /**
  47.              * 只要消息没有投递给指定的队列,就触发这个失败回调
  48.              * @param message   投递失败的消息详细信息
  49.              * @param replyCode 回复的状态码
  50.              * @param replyText 回复的文本内容
  51.              * @param exchange  当时这个消息发给哪个交换机
  52.              * @param routingKey 当时这个消息用哪个路由键
  53.              */
  54.             @Override
  55.             public void returnedMessage(Message message, int replyCode, String replyText, String exchange, String routingKey) {
  56.                 //报错误了。修改数据库当前消息的状态->错误。
  57.                 System.out.println("Fail Message["+message+"]==>replyCode["+replyCode+"]==>replyText["+replyText+"]===>exchange["+exchange+"]===>routingKey["+routingKey+"]");
  58.             }
  59.         });
  60.     }
复制代码


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。




欢迎光临 IT评测·应用市场-qidao123.com (https://dis.qidao123.com/) Powered by Discuz! X3.4