异步通信rabbitmq——消息重试

目标:

    利用RabbitMQ实现消息重试和失败处理,实现可靠的消费消费。在消息消费异常时,自动延时将消息重试,当重试超过一定次数后,则列为异常消息,等待后续特殊处理。

准备:

    TTL:Time-To-Live,通过给消息、队列设置过期时间(单位:毫秒),来控制消息、队列的生命周期。在达到时间后,消息会变成dead message。

    Dead Letter Exchanges:同普通的exchange无区别

    消息重制本质是通过消息转发来实现的。消息转发的触发是:

  • rejected - the message was rejected with requeue=false,
  • expired - the TTL of the message expired; or
  • maxlen - the maximum allowed queue length was exceeded.

这里,我们使用expired来实现。给消息设置TTL,到期后消息未被消费,则会变成dead messager,转发到dead letter exchange。

流程图:


实现:

1、创建三个exchange。没有特殊要求


2、创建三个queue。


clickQueue@retry作为重试队列,需要特殊处理:

x-dead-letter-exchange: clickExchange
x-dead-letter-routing-key: clickKey

x-message-ttl: 30000

3、处理代码

public void retry() throws IOException {
    //消息消费
    GetResponse getResponse = null;
    try {
        getResponse = rabbitUtil.fetch(DQConstant.CLICK_QUEUE_NAME, false);
        /**
         * 业务处理
         */
        throw new RuntimeException("错粗了");
    } catch (Exception e) {
        if(null != getResponse) {
            long retryCount = getRetryCount(getResponse.getProps());
            if(retryCount > 3) {
                //重试超过3次的,直接存入失败队列
                AMQP.BasicProperties properties = getResponse.getProps();
                Map<String, Object> headers = properties.getHeaders();
                if(null == headers) {
                    headers = new HashMap<>();
                }
                properties.builder().headers(headers);
                rabbitUtil.send(DQConstant.CLICK_FAILED_EXCHANGE_NAME, DQConstant.CLICK_FAILED_ROUTING_KEY, properties, getResponse.getBody());
            } else {
                //重试不超过3次的,加入到重试队列
                AMQP.BasicProperties properties = getResponse.getProps();
                Map<String, Object> headers = properties.getHeaders();
                if(null == headers) {
                    headers = new HashMap<>();
                }
                properties.builder().headers(headers);
                rabbitUtil.send(DQConstant.CLICK_RETRY_EXCHANGE_NAME, DQConstant.CLICK_RETRY_ROUTING_KEY, properties, getResponse.getBody());
            }
        }
    }
    if(null != getResponse) {
        rabbitUtil.ack(getResponse);
    }
}

private long getRetryCount(AMQP.BasicProperties properties) {
    long retryCount = 0;
    Map<String, Object> headers = properties.getHeaders();
    if(null != headers) {
        if(headers.containsKey("x-death")) {
            List<Map<String, Object>> deathList = (List<Map<String, Object>>) headers.get("x-death");
            if(!deathList.isEmpty()) {
                Map<String, Object> deathEntry = deathList.get(0);
                retryCount = (Long)deathEntry.get("count");
            }
        }
    }
    return retryCount;
}

4、x-death的使用:message在转换成dead letter时,会在其header里添加一个名为x-death的数组。数组元素就是一次dead lettering event的记录。包含count:消息几次变成了dead letter。


总结:

此处只是本人的拙见。如有更好的提议,欢迎拍砖。

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值