《RabbitMQ系列教程-第十三章-消息的幂等性》

追求适度,才能走向成功;人在顶峰,迈步就是下坡;身在低谷,抬足既是登高;弦,绷得太紧会断;人,思虑过度会疯;水至清无鱼,人至真无友,山至高无树;适度,不是中庸,而是一种明智的生活态度。

导读:本篇文章讲解 《RabbitMQ系列教程-第十三章-消息的幂等性》,希望对大家有帮助,欢迎收藏,转发!站点地址:www.bmabk.com,来源:原文

教程说明



消息的幂等性

13.1 幂等简介

幂等性:用户对于同一操作发起的一次请求或者多次请求的结果是一致的,不会因为多次点击而产生了副作用,例如用户在下订单时,由于网络卡顿等原因多次点击下单按钮,但最终订单只生成了一个,并不会多次下订单;

消息幂等性:即消息无论被发送多少次,最终只会消费一次。保证消息不会重复被消费多次;

13.1.1 方案

实现消息的幂等性通常有两种方案来解决:全局消息ID、Redis标识,两种方案从原理上讲都是一样的;

  • 全局消息ID:发送消息时给消息分配一个全局ID,每次消费消息时判断该ID是否存在过,如果存在过则已经消费过,如果不存在则说明是第一次消费;
  • Redis标识:发送消息时,给消息分配一个全局的唯一ID,消费消息时,将id与消息以K-V的形式写入Redis<id,meessage>,如果能写入成功代表第一次消费消息,如果写入不成功代表消费已经被消费过了

13.1.2 解决消息幂等性

搭建SpringBoot整合RabbitMQ工程

我们采用的是Redis的setnx命令来解决消息幂等性问题;

application.yml:

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

启动类:

package com.lscl.rabbitmq;

import org.springframework.amqp.core.Queue;
import org.springframework.amqp.core.QueueBuilder;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.Bean;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer;
import org.springframework.data.redis.serializer.StringRedisSerializer;

@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class);
    }

    @Bean
    public RedisTemplate redisTemplate(RedisTemplate redisTemplate){

        // value可见
        redisTemplate.setValueSerializer(new GenericJackson2JsonRedisSerializer());

        // key可见
        redisTemplate.setKeySerializer(new StringRedisSerializer());

        return redisTemplate;
    }

    // 定义测试队列
    @Bean
    public Queue testQueue(){
        return QueueBuilder.durable("test_queue").build();
    }
}

监听器:

package com.lscl.rabbitmq.listener;

import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;

@Component
public class TestListener {

    @Autowired
    private RedisTemplate redisTemplate;

    @RabbitListener(queues = "test_queue")
    public void test_queue_confirm(Message message) {

        String messageId = message.getMessageProperties().getMessageId();

        if (null == messageId) {

            System.out.println("消息id为null!");
            return;
        }

        if (redisTemplate.opsForValue().setIfAbsent(messageId, new String(message.getBody()))) {

            // 代表第一次消费消息
            System.out.println("消息消费成功: " + new String(message.getBody()));
        } else {
            System.out.println("消息已经被消费过了!");
        }
    }
}

消息发送端:

package com.lscl.rabbitmq.controller;

import org.springframework.amqp.core.Message;
import org.springframework.amqp.core.MessageProperties;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class TestController {

    @Autowired
    private RabbitTemplate rabbitTemplate;

    @GetMapping("/send/{messageId}")
    public String addOrders(@PathVariable String messageId) {

        MessageProperties prop = new MessageProperties();
        prop.setMessageId(messageId);
        Message message = new Message("测试幂等性".getBytes(), prop);

        // 发送消息
        rabbitTemplate.convertAndSend("test_queue", message);

        return messageId;
    }
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/131786.html

(0)
飞熊的头像飞熊bm

相关推荐

发表回复

登录后才能评论
极客之音——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!