Rabbitmq消费者保证幂等性

如图:

 示例代码:

pom文件添加依赖

<dependency>
   <groupId>org.springframework.boot</groupId>
   <artifactId>spring-boot-starter-amqp</artifactId>
</dependency>

application.yml配置rabbitmq连接

spring:
  rabbitmq:
    host: 127.0.0.1
    port: 5672
    username: guest
    password: guest
    virtual-host: /guoVirtualHost

    listener:
      simple:
        retry:
#          开启消费者(出现异常)会进行重试
          enabled: true
#          默认重试无限次,所以需指定重试次数
          max-attempts: 5
#          重试间隔时间
          initial-interval: 3000
#          保证消费者会消费消息,手动确认
        acknowledge-mode: manual

#定义交换机、路由、队列名称
guoguo:
  order:
    exchange: guoguo_order_exchange
    routingKey: guoguo.order
    queue: guoguo_order_queue

定义RabbitmqConfig配置类,Exchange,Routingkey,Queue

@Component
public class OrderConfig {
    
    //交换机名称 
    @Value("${guoguo.order.exchange}")
    private String orderExchange;
    //路由key
    @Value("${guoguo.order.routingKey}")
    private String orderRoutingKey;
    //队列名称
    @Value("${guoguo.order.queue}")
    private String orderQueue;

    @Bean
    public DirectExchange orderExchange(){
       return new DirectExchange(orderExchange);
    }
    @Bean
    public Queue orderQueue(){
        return new Queue(orderQueue,true,false,false);
    }
    @Bean
    public Binding orderBinding(Queue orderQueue,DirectExchange orderExchange){
        //方式1:Queue orderQueue,DirectExchange orderExchange 会去ioc容器中找到Bean
        return BindingBuilder.bind(orderQueue).to(orderExchange).with(orderRoutingKey);
        //方式2:直接调用方法
   //return BindingBuilder.bind(orderQueue()).to( orderExchange()).with(orderRoutingKey);
    }

定义发送消息工具类:

@Component
@Slf4j
public class SendMessageUtlis {

    @Autowired
    RabbitTemplate amqpTemplate;
    
    @Value("${guoguo.order.exchange}")
    private String orderExchange;

    @Value("${guoguo.order.routingKey}")
    private String orderRoutingKey;

    public void sendMessage(Order order){
        amqpTemplate.convertAndSend(orderExchange,orderRoutingKey,order,message -> {
            return message;
        });
    }

发送消息:

  @RequestMapping("/addOrder")
    public String addOrder(){
        String quanjuId = System.currentTimeMillis()+ "";
        //生产者投递消息时携带全局id发送到mq服务器端
        Order order = new Order(1, "mq演示", 66, "男",quanjuId);
        log.info("全局id:"+quanjuId);
        sendMessage.sendMessage(order );
        //返回全局id给客户端
        return quanjuId;
    }

消费者监听消息:

    @Autowired
    OrderDao orderDao;

    @RabbitListener(queues = "guoguo_order_queue")
    public void process(Order order, Message message,Channel channel) throws IOException {
        if (StringUtils.isEmpty(order.getQuanjuId())){
            return;
        }
        //拿到全局id
        Order orders = orderDao.getQuanjuId(order.getQuanjuId());
        if (order!= null){
            log.info("已经被消费");
            //如果不为null,则已经被消费过,也需要告诉mq服务器端,将该消息删除
            channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);
            return;
        }
        int result = orderDao.saveOrder(order);
        log.info("插入成功");
        channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);
    }

  • 4
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
当我们使用 RabbitMQ 作为消息队列时,消费者需要处理重复消费的情况。实现幂等性处理可以避免重复消费带来的问题。以下是一个示例,演示如何使用 Redis 实现 RabbitMQ 消费者幂等性处理: 1. 在消费者应用中引入 Redis 的依赖。 2. 在消费者处理消息前,先查询 Redis 中是否已经处理过该消息。如果已经处理过,则不再处理;如果未处理,则处理消息,并将消息处理的标识存储到 Redis 中。 示例代码: ```python import redis import pika # Redis 配置 redis_host = 'localhost' redis_port = 6379 redis_db = 0 redis_key_prefix = 'rabbitmq_msg_processed:' # 连接 Redis redis_conn = redis.Redis(host=redis_host, port=redis_port, db=redis_db) # RabbitMQ 配置 rabbitmq_host = 'localhost' rabbitmq_port = 5672 rabbitmq_user = 'guest' rabbitmq_password = 'guest' rabbitmq_exchange = 'example_exchange' rabbitmq_queue = 'example_queue' rabbitmq_routing_key = 'example_routing_key' # 连接 RabbitMQ credentials = pika.PlainCredentials(rabbitmq_user, rabbitmq_password) connection = pika.BlockingConnection(pika.ConnectionParameters(host=rabbitmq_host, port=rabbitmq_port, credentials=credentials)) channel = connection.channel() # 声明 exchange 和 queue channel.exchange_declare(exchange=rabbitmq_exchange, exchange_type='direct', durable=True) channel.queue_declare(queue=rabbitmq_queue, durable=True) channel.queue_bind(queue=rabbitmq_queue, exchange=rabbitmq_exchange, routing_key=rabbitmq_routing_key) # 定义消息处理函数 def callback(ch, method, properties, body): msg_id = properties.message_id redis_key = redis_key_prefix + msg_id # 查询 Redis 中是否已经处理过该消息 if not redis_conn.exists(redis_key): # 处理消息 print('Received message: %r' % body) # 将消息处理的标识存储到 Redis 中 redis_conn.set(redis_key, 1) print('Processed message: %r' % body) else: print('Message already processed: %r' % body) ch.basic_ack(delivery_tag=method.delivery_tag) # 消费消息 channel.basic_qos(prefetch_count=1) channel.basic_consume(queue=rabbitmq_queue, on_message_callback=callback) print('Waiting for messages...') channel.start_consuming() ``` 在上述示例中,我们定义了一个 Redis 的连接对象 `redis_conn`,并在消息处理函数 `callback` 中使用该对象查询并存储消息处理的标识。如果消息处理过,则不再处理;如果未处理,则处理消息并存储标识。这样可以保证每条消息只被处理一次,避免了重复消费的问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值