消息的重发以及补充策略

欢迎大家关注我的公众号【老周聊架构】,Java后端主流技术栈的原理、源码分析、架构以及各种互联网高并发、高性能、高可用的解决方案。

一、MQ消息发送

在这里插入图片描述

1、发送端MQ-client(消息生产者:Producer)将消息发送给MQ-server;

2、MQ-server将消息落地;

3、MQ-server回ACK给MQ-client(Producer);

4、MQ-server将消息发送给消息接受端MQ-client(消息消费者:Customer);

5、MQ-client(Customer)消费接受到消息后发送ACK给MQ-server;

6、MQ-server将落地消息删除

二、消息重复发送原因

为了保证消息必达,MQ使用了消息超时、重传、确认机制。使得消息可能被重复发送,如上图中,由于网络不可达原因:3和5中断,可能导致消息重发。消息生产者a收不到MQ-server的ACK,重复向MQ-server发送消息。MQ-server收不到消息消费者c的ACK,重复向消息消费者c发消息。

三、消息重复发送产生的后果

举个例子:购买会员卡,上游支付系统负责给用户扣款,下游系统负责给用户发卡,通过MQ异步通知。不管是上半场的ACK丢失,导致MQ收到重复的消息,还是下半场ACK丢失,导致购卡系统收到重复的购卡通知,都可能出现,上游扣了一次钱,下游发了多张卡。

四、MQ内部如何做到幂等性的

对于每条消息,MQ内部生成一个全局唯一、与业务无关的消息ID:inner-msg-id。当MQ-server接收到消息时,先根据inner-msg-id判断消息是否重复发送,再决定是否将消息落地到DB中。这样,有了这个inner-msg-id作为去重的依据就能保证一条消息只能一次落地到DB。

五、消息消费者应当如何做到幂等性

1、对于非幂等性业务且要求实现幂等性业务:生成一个唯一ID标记每一条消息,将消息处理成功和去重日志通过事物的形式写入去重表。

2、对于非幂等性业务可不实现幂等性的业务:权衡去重所花的代价决定是否需要实现幂等性,如:购物会员卡成功,向用户发送通知短信,发送一次或者多次影响不大。不做幂等性可以省掉写去重日志的操作。

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
在 RabbitMQ 中,当消息发送失败时,可以使用以下两种方式进行重发: 1. 自动重试:可以在发送消息时设置重试次数和重试间隔时间,如果消息发送失败,则自动重试发送。这种方式可以通过 RabbitMQ 的 Retry 插件来实现。 2. 手动重发:如果消息发送失败或者未被确认,则可以手动重新发送消息。具体实现方式是在发送消息时,将消息设置为可重发状态,并将其保存在一个队列中,然后按照一定的规则从队列中取出消息进行重发。 以下是手动重发消息的代码示例: ```java ConnectionFactory factory = new ConnectionFactory(); factory.setHost("localhost"); Connection connection = factory.newConnection(); Channel channel = connection.createChannel(); String exchangeName = "test-exchange"; String routingKey = "test-routing-key"; String message = "Hello RabbitMQ!"; try { channel.basicPublish(exchangeName, routingKey, null, message.getBytes()); } catch (IOException e) { // 发送消息失败,将消息设置为可重发状态,并保存到队列中 channel.basicPublish(exchangeName, routingKey, MessageProperties.PERSISTENT_TEXT_PLAIN, message.getBytes()); } // 从队列中取出消息进行重发 GetResponse response = channel.basicGet(queueName, false); if (response != null) { try { channel.basicPublish(exchangeName, routingKey, null, response.getBody()); } catch (IOException e) { // 重发消息失败,将消息再次设置为可重发状态,并保存到队列中 channel.basicPublish(exchangeName, routingKey, MessageProperties.PERSISTENT_TEXT_PLAIN, response.getBody()); } finally { channel.basicAck(response.getEnvelope().getDeliveryTag(), false); } } channel.close(); connection.close(); ``` 在上述示例中,如果发送消息失败,则将消息设置为可重发状态,并保存到队列中。然后从队列中取出消息进行重发,如果重发失败,则再次将消息设置为可重发状态并保存到队列中。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

老周聊架构

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值