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版本


    org.apache.rocketmq
    rocketmq-spring-boot-starter
    2.2.2

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,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;
        });
    }
}

 

你可能感兴趣的:(java-rocketmq,rocketmq,redis)