RabbitMQ| 解决消息幂等性问题

消费者出现业务逻辑问题,自动补偿机制

场景演示:

/**
 * @author 孙一鸣 on 2020/3/15
 */
@Service
@RabbitListener(bindings = @QueueBinding(
        value = @Queue(value = "sym"),
        exchange = @Exchange(value = "exchange.direct",type = "direct"),
        key = "sym"
))
public class rabbitService {
    @RabbitListener(queues = "sym")
    public void recive (Map map){
        System.out.println(map.get("data"));
        int i=1/0;
    }
}

@RabbitListener 底层 使用Aop进行拦截,如果程序没有抛出异常,自动提交事务
// 如果Aop使用异常通知拦截 获取异常信息的话,自动实现补偿机制 ,该消息会缓存到rabbitmq服务器端进行存放,一直重试到不抛异常为准。

补偿(重试机制) 队列服务器 发送补偿请求

修改重试机制策略 一般默认情况下 间隔3秒重试一次
在这里插入图片描述
发送消息者发送消息,消费者消费消息时出现异常,此时队列重试3次
出现3次日志显示

在这里插入图片描述

调用第三方接口,如果出现异常,调用重试机制

	@RabbitListener(queues = "fanout_email_queue")
	public void process(Message message, @Headers Map<String, Object> headers, Channel channel) throws Exception {
		String messageId = message.getMessageProperties().getMessageId();
		String msg = new String(message.getBody(), "UTF-8");
		System.out.println("邮件消费者获取生产者消息msg:" + msg + ",消息id:" + messageId);
		// 重试机制都是间隔性

		JSONObject jsonObject = JSONObject.parseObject(msg);
		String email = jsonObject.getString("email");
		String emailUrl = "http://127.0.0.1:8083/sendEmail?email=" + email;
		System.out.println("邮件消费者开始调用第三方邮件服务器,emailUrl:" + emailUrl);
		JSONObject result = HttpClientUtils.httpGet(emailUrl);
		// 如果调用第三方邮件接口无法访问,如何实现自动重试.
		if (result == null) {
			throw new Exception("调用第三方邮件服务器接口失败!");
		}
		System.out.println("邮件消费者结束调用第三方邮件服务器成功,result:" + result + "程序执行结束");
		// 手动ack
		Long deliveryTag = (Long) headers.get(AmqpHeaders.DELIVERY_TAG);
		// 手动签收
		channel.basicAck(deliveryTag, false);

	}
	// 默认是自动应答模式

在这里插入图片描述

MQ重试机制需要注意的问题

如何合适选择重试机制:

  • 情况1: 消费者获取到消息后,调用第三方接口,但接口暂时无法访问,是否需要重试? (需要重试机制
  • 情况2: 消费者获取到消息后,抛出数据转换异常,是否需要重试?(不需要重试机制)需要发布进行解决

消费者如果保证消息幂等性,不被重复消费
产生原因:网络延迟传输中,消费出现异常或者是消费延迟消费,会造成MQ进行重试补偿,在重试过程中,可能会造成重复消费。

解决办法:
①使用全局MessageID判断消费方使用同一个,解决幂等性。


@Component
public class FanoutProducer {
	@Autowired
	private AmqpTemplate amqpTemplate;

	public void send(String queueName) {
		JSONObject jsonObject = new JSONObject();
		jsonObject.put("email", "644064779");
		jsonObject.put("timestamp", System.currentTimeMillis());
		String jsonString = jsonObject.toJSONString();
		System.out.println("jsonString:" + jsonString);
		// 生产者发送消息的时候需要设置消息id

		Message message = MessageBuilder.withBody(jsonString.getBytes())
				.setContentType(MessageProperties.CONTENT_TYPE_JSON).setContentEncoding("utf-8")
				.setMessageId(UUID.randomUUID() + "").build();

		amqpTemplate.convertAndSend(queueName, message);
	}
}

②或者使用业务逻辑保证唯一(比如订单号码)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章