RocketMq普通消息,死信队列,消息幂等性(redis)

1 介绍

RocketMQ作为一款纯java、分布式、队列模型的开源消息中间件,支持事务消息、顺序消息、批量消息、定时消息、消息回溯等。

 

1.1 RocketMQ 特点

  • 支持发布/订阅(Pub/Sub)和点对点(P2P)消息模型

  • 在一个队列中可靠的先进先出(FIFO)和严格的顺序传递 (RocketMQ可以保证严格的消息顺序,而ActiveMQ无法保证)

  • 支持拉(pull)和推(push)两种消息模式

    pull其实就是消费者主动从MQ中去拉消息,而push则像rabbit MQ一样,是MQ给消费者推送消息。但是RocketMQ的push其实是基于pull来实现的。
    它会先由一个业务代码从MQ中pull消息,然后再由业务代码push给特定的应用/消费者。其实底层就是一个pull模式

  • 单一队列百万消息的堆积能力 (RocketMQ提供亿级消息的堆积能力,这不是重点,重点是堆积了亿级的消息后,依然保持写入低延迟)

  • 支持多种消息协议,如 JMS、MQTT 等

  • 分布式高可用的部署架构,满足至少一次消息传递语义(RocketMQ原生就是支持分布式的,而ActiveMQ原生存在单点性)

  • 提供 docker 镜像用于隔离测试和云集群部署

  • 提供配置、指标和监控等功能丰富的 Dashboard

新建springboot工程并,导入依赖,本文选用rocketmq2.2.2版本

<dependency>
    <groupId>org.apache.rocketmq</groupId>
    <artifactId>rocketmq-spring-boot-starter</artifactId>
    <version>2.2.2</version>
</dependency>

YML配置文件

rocketmq:
  name-server: 192.168.198.129:9876
  producer:
    group: group1

 消息提供者

    @GetMapping("tset")
    public String send(){
        //尝试发送消息
        rocketMQTemplate.convertAndSend("topic1","smoky哥哥呀");
//        log.info("尝试发送同步消息:{}",syncSend);
        return "success";
    }

 

消息消费者

@Component
@RocketMQMessageListener(topic = "topic1",consumerGroup = "group1")
@Slf4j
public class ReceiveConsumer implements RocketMQListener<String>,RocketMQPushConsumerLifecycleListener{

    @Autowired
    StringRedisTemplate stringRedisTemplate;

    @Override
    public void onMessage(String s) {
        log.info("接收到MQ消息:{}",s);
    }

    @Override
    public void prepareStart(DefaultMQPushConsumer defaultMQPushConsumer) {
        defaultMQPushConsumer.registerMessageListener((MessageListenerConcurrently) (list, consumeConcurrentlyContext) -> {
            MessageExt messageExt = list.get(0);
            //尝试获取消息体内容
            String body = new String(messageExt.getBody());
            //尝试获取消息id
            String msgId = messageExt.getMsgId();
            log.info("获取到消息id为:{}",msgId);
            //尝试获取重试次数
            int reconsumeTimes = messageExt.getReconsumeTimes();
            log.info("获取到当前消息重试次数:{}",reconsumeTimes);
            try {
                int i = 10/0;
                //接口幂等性
                Boolean rocketmq = stringRedisTemplate.opsForHash().hasKey("ROCKETMQ", msgId);
                if (!rocketmq) {
                    log.info("接收到ACK消息:{}", body);
                    //存入redis日志表,代表消费成功
                    stringRedisTemplate.opsForHash().put("ROCKETMQ",msgId,"1");
                }
            } catch (Exception e) {
                e.printStackTrace();
                log.info("异常捕获,尝试ACK重试机制");
                if (reconsumeTimes>=3){
                    //加入死信队列
                    log.info("业务逻辑:加入死信队列做,做事后补偿");
                    return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
                }
                //发起重试
                log.info("尝试发起ACK重试");
                return ConsumeConcurrentlyStatus.RECONSUME_LATER;
            }
            return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
        });
    }
}

 

 

 

  • 5
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

是Smoky呢

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

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

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

打赏作者

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

抵扣说明:

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

余额充值