RabbitMQ消息重复消费场景及解决方案

前言

上一篇文章介绍了springboot如何整合RabbitMQ:springboot集成RabbitMQ_龙池小生的博客-CSDN博客

这里介绍一下RabbitMQ重复消费的场景,以及如何解决消息重复消费的问题。

注:本文只做粗略逻辑实现借鉴,实际业务场景需根据实际情况再做细化处理。

目录

消息重复消费:

MQ的一条消息被消费者消费了多次:

重复消费场景重现测试:

如何解决消息重复消费的问题:

编码:

解决消息重复消费测试:


消息重复消费:

什么是消息重复消费?首先我们来看一下消息的传输流程。消息生产者--》MQ--》消息消费者;消息生产者发送消息到MQ服务器,MQ服务器存储消息,消息消费者监听MQ的消息,发现有消息就消费消息。

所以消息重复也就出现在两个阶段1、生产者多发送了消息给MQ;2、MQ的一条消息被消费者消费了多次。第一种场景很好控制,只要保证消息生成者不重复发送消息给MQ即可。我们着重来看一下第二个场景。

MQ的一条消息被消费者消费了多次

在保证MQ消息不重复的情况下,消费者消费消息成功后,在给MQ发送消息确认的时候出现了网络异常(或者是服务中断),MQ没有接收到确认,此时MQ不会将发送的消息删除,
为了保证消息被消费,当消费者网络稳定后,MQ就会继续给消费者投递之前的消息。这时候消费者就接收到了两条一样的消息。

重复消费场景重现测试

1、消息发送者发送1万条消息给MQ:

    @GetMapping("/rabbitmq/sendToClient")
    public String sendToClient() {
        String message = "server message sendToClient";
        for (int i = 0; i < 10000; i++) {
            amqpTemplate.convertAndSend("queueName3",message+": "+i);

        }
        return message;
    }

启动消息发送服务,调用接口发送消息,mq成功收到1万条消息。

2、消费者监听消费消息:

    @RabbitListener(queues = "queueName3")//发送的队列名称     @RabbitListener注解到类和方法都可以
    @RabbitHandler
    public void receiveMessage(String message) {
        System.out.println("接收者2--接收到queueName3队列的消息为:"+message);
    }

启动消费者服务,然后中断消费服务,此时消费到了第7913个消息:

此时查看MQ的消息,现在MQ队列中应该还有2087个消息,但还有2088个消息,说明最后一个消息被消费了没有被MQ服务确认。

再次启动消费者服务,消息从第7913个消息开始消费,而不是第7914个消息

如何解决消息重复消费的问题:

为了保证消息不被重复消费,首先要保证每个消息是唯一的,所以可以给每一个消息携带一个全局唯一的id,流程如下

1、消费者监听到消息后获取id,先去查询这个id是否存中

2、如果不存在,则正常消费消息,并把消息的id存入 数据库或者redis中(下面的编码示例使用redis)

3、如果存在则丢弃此消息

编码:

消息生产者服务

    /**
     * @Description:  发送消息 模拟消息重复消费
     *      消息重复消费情景:消息生产者已把消息发送到mq,消息消费者在消息消费的过程中突然因为网络原因或者其他原因导致消息消费中断
     *      消费者消费成功后,在给MQ确认的时候出现了网络波动,MQ没有接收到确认,
     *      为了保证消息被消费,MQ就会继续给消费者投递之前的消息。这时候消费者就接收到了两条一样的消息
     * @param:
     * @return: java.lang.String
     * @Author: chenping
     * @Date: 2021/3/5 17:25
     */
    @GetMapping("/rabbitmq/sendMsgNoRepeat")
    public String sendMsgNoRepeat() {
        String message = "server message sendMsgNoRepeat";
        for (int i = 0; i <10000 ; i++) {
            Message msg = MessageBuilder.withBody((message+"--"+i).getBytes()).setMessageId(UUID.randomUUID()+"").build();
            amqpTemplate.convertAndSend("queueName4",msg);
        }
        return message;
    }

消息消费者服务

方案1:将id存入string中(单消费者场景):

这样一个队列,redis数据只有一条,每次消息过来都覆盖之前的消息,但是消费者多的情况不适用,可能会存在问题--一个消息被多个消费者消费

    @RabbitListener(queues = "queueName4")//发送的队列名称     @RabbitListener注解到类和方法都可以
    @RabbitHandler
    public void receiveMessage(Message message) throws UnsupportedEncodingException {
        String messageId = message.getMessageProperties().getMessageId();
        String msg = new String(message.getBody(),"utf-8");

        String messageRedisValue = redisUtil.get("queueName4","");
        if (messageRedisValue.equals(messageId)) {
            return;
        }
        System.out.println("消息:"+msg+", id:"+messageId);

        redisUtil.set("queueName4",messageId);//以队列为key,id为value
    }

方案2:将id存入list中(多消费者场景)

这个方案可以解决多消费者的问题,但是随着mq的消息增加,redis数据越来越多,需要去清除redis数据。

    @RabbitListener(queues = "queueName4")//发送的队列名称     @RabbitListener注解到类和方法都可以
    @RabbitHandler
    public void receiveMessage1(Message message) throws UnsupportedEncodingException {
        String messageId = message.getMessageProperties().getMessageId();
        String msg = new String(message.getBody(),"utf-8");

        List<String> messageRedisValue = redisUtil.lrange("queueName4");
        if (messageRedisValue.contains(messageId)) {
            return;
        }
        System.out.println("消息:"+msg+", id:"+messageId);

        redisUtil.lpush("queueName4",messageId);//存入list
    }

方案3:将id以key值增量存入string中并设置过期时间:

以消息id为key,消息内容为value存入string中,设置过期时间(可承受的redis服务器异常时间,比如设置过期时间为10分钟,如果redis服务器断了20分钟,那么未消费的数据都会丢了)

    @RabbitListener(queues = "queueName4")//发送的队列名称     @RabbitListener注解到类和方法都可以
    @RabbitHandler
    public void receiveMessage2(Message message) throws UnsupportedEncodingException {
        String messageId = message.getMessageProperties().getMessageId();
        String msg = new String(message.getBody(),"utf-8");

        String messageRedisValue = redisUtil.get(messageId,"");
        if (msg.equals(messageRedisValue)) {
            return;
        }
        System.out.println("消息:"+msg+", id:"+messageId);

        redisUtil.set(messageId,msg,10L);//以id为key,消息内容为value,过期时间10分钟
    }

 

解决消息重复消费测试:

首先,启动消息生成服务,发送一万条消息

启动消息消费服务,然后中断服务,消费了1934条消息

查看未被消费的消息条数为8067条,多了一条(10000-1934=8066 )

再次启动消费者服务,消费者舍弃了已被消费的第1934条消息

  • 40
    点赞
  • 115
    收藏
    觉得还不错? 一键收藏
  • 20
    评论
解决RabbitMQ消息重复消费的方法有多种。一种方法是在消费消费消息成功后,在给MQ发送消息确认的时候出现网络异常或服务中断的情况下,MQ不会将发送的消息删除,而是会继续给消费者投递之前的消息,导致消费者接收到了两条一样的消息。为了解决这个问题,可以在消费者端使用幂等性来处理消息,即使接收到重复消息,也能保证最终的结果是一致的。\[1\]\[2\] 另一种方法是在消费者监听消费消息时,使用消息的唯一标识来判断是否已经消费过该消息。可以在消费者端记录已经消费过的消息的标识,当接收到重复消息时,先判断该消息的标识是否已经存在,如果存在则不进行消费,避免重复消费。\[3\] 除了以上两种方法,还可以使用消息的幂等性保证机制,即在消息的生产者端设置消息的唯一标识,并在消费者端进行幂等性校验,如果消息的唯一标识已经存在,则不进行消费。这样可以避免重复消费的问题。 总之,解决RabbitMQ消息重复消费的方法包括使用幂等性处理消息、记录已经消费过的消息的标识以及使用消息的唯一标识进行幂等性校验等。根据具体的业务需求和场景,选择合适的方法来解决重复消费的问题。 #### 引用[.reference_title] - *1* *2* *3* [RabbitMQ消息重复消费场景解决方案](https://blog.csdn.net/chenping1993/article/details/114580954)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论 20
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值