引入RabbitMQ

前置条件

docker 安装 mq

docker run \
 -e RABBITMQ_DEFAULT_USER=dudu \
 -e RABBITMQ_DEFAULT_PASS=123456 \
 -v mq-plugins:/plugins \
 --name mq \
 --hostname mq \
 -p 15672:15672 \
 -p 5672:5672 \
 --network hmall \
 -d \
 rabbitmq:3.8-management

可能会出现:docker: Error response from daemon: network hmall not found.
原因是在容器启动时,所需的网络环境没有正确配置。

检查网络列表

docker network ls

创建所需网络

docker network create hmall

运行容器时指定网络

docker run -d --net=hmall rabbitmq:3.8-management

重新启动容器

docker restart mq

新建初始工程

父工程引入依赖

  		<!--AMQP依赖,包含RabbitMQ-->
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-amqp</artifactId>
		</dependency>

publisher 和 consumer 引入 yml 配置

spring:
  rabbitmq:
    host: 192.168.64.100 # 你的虚拟机IP
    port: 5672 # 端口
    virtual-host: /dudu # 虚拟主机
    username: dudu # 用户名
    password: 123456 # 密码

###基本消息模型

新建虚拟主机

新建 base.queue 队列

publisher 测试类发送消息

@SpringBootTest
public class BaseTest {
	@Autowired
	private RabbitTemplate rabbitTemplate;
	@Test
	public void testSendMessage(){
		//队列名称
		String queueName = "base.queue";
		//消息
		String message = "基本消息模型测试";
		//发送消息
		rabbitTemplate.convertAndSend(queueName,message);
	}
}

consumer 配置监听消息

@Component
public class RabbitMQListener {
	// 监听基本消息模型 base.queue队列
	@RabbitListener(queues = "base.queue")
	public void baseListener(String msg) {
		System.out.println("base.queue接收到消息:" + msg);
	}
}

work 消息模型

  • 多个消费者绑定到一个队列,同一条消息只会被一个消费者处理
  • 通过设置 prefetch 来控制消费者预取的消息数量

同基本消息模型一样新建队列:work.queue

publisher 测试类发送消息

@SpringBootTest
public class BaseTest {
	@Autowired
	private RabbitTemplate rabbitTemplate;
	@Test
	public void testSendWorkMessage() {
		for (int i = 1; i <= 10; i++) {
			String message = "测试message" + i;
			rabbitTemplate.convertAndSend("work.queue",message);
		}
	}
}

consumer 配置监听消息

@Component
public class RabbitMQListener {
	// 监听 work 消息模型 work.queue队列
	@RabbitListener(queues = "work.queue")
	public void workListener1(String msg) {
		System.out.println("消费者一接收到work.queue的消息:"+ msg);
	}
	@RabbitListener(queues = "work.queue")
	public void workListener2(String msg) {
		System.err.println("消费者二接收到work.queue的消息:"+ msg);
	}
}

测试


默认是消费者平分消息,并没有考虑到消费者的处理能力。可能会存在一个消费者空闲,一个消费者忙,没有充分的利用消费者。

在 spring 中有一个简单的配置,可以解决这个问题。我们修改 consumer 服务的 application.yml 文件,修改配置:

spring:
  rabbitmq:
    host: 192.168.64.100 # 你的虚拟机IP
    port: 5672 # 端口
    virtual-host: /dudu # 虚拟主机
    username: dudu # 用户名
    password: 123456 # 密码
    listener:
      simple:
        prefetch: 1 # 每次只能获取一条消息,处理完成才能获取下一个消息

再次测试

Fanout 交换机消息模型(广播)

  • 接收 publisher 发送的消息
  • 将消息按照规则路由到与之绑定的队列
  • 不能缓存消息,路由失败,消息丢失
  • FanoutExchange 的会将消息路由到每个绑定的队列

新建交换机和队列

将队列绑定到交换机

publisher 测试类发送消息

@SpringBootTest
public class BaseTest {
	@Autowired
	private RabbitTemplate rabbitTemplate;
	@Test
	public void testSendFanoutMessage() {
		String exchangeName = "dudu.fanout";
		String message = "测试Fanout消息模型";
		rabbitTemplate.convertAndSend(exchangeName,"",message);
  }
}

consumer 配置监听消息

@Component
public class RabbitMQListener {
	// 监听 fanout 交换机消息模型(广播) work.queue队列
	@RabbitListener(queues = "fanout.queue1")
	public void fanoutListener1(String msg) {
		System.out.println("消费者一接收到 fanout.queue1 的消息:"+ msg);
	}
	@RabbitListener(queues = "fanout.queue2")
	public void fanoutListener2(String msg) {
		System.err.println("消费者二接收到 fanout.queue2 的消息:"+ msg);
	}

测试

Direct 交换机消息模型(发布-订阅)

  • Fanout 交换机将消息路由给每一个与之绑定的队列
  • Direct 交换机根据 RoutingKey 判断路由给哪个队列
  • 如果多个队列具有相同的 RoutingKey,则与 Fanout 功能类似

新建交换机,队列,绑定路由 key

publisher 测试类发送消息

@SpringBootTest
public class BaseTest {
	@Autowired
	private RabbitTemplate rabbitTemplate;
	@Test
	public void testSendDirectMessage() {
		String exchangeName = "dudu.direct";
		String message = "测试Direct消息模型";
		rabbitTemplate.convertAndSend(exchangeName,"red",message+"红色消息");
		rabbitTemplate.convertAndSend(exchangeName,"blue",message+"蓝色消息");
		rabbitTemplate.convertAndSend(exchangeName,"yellow",message+"黄色消息");
	}
}

consumer 配置监听消息

@Component
public class RabbitMQListener {
	// 监听 direct 交换机消息模型
	@RabbitListener(queues = "direct.queue1")
	public void directListener1(String msg) {
		System.out.println("消费者一接收到 direct.queue1 的消息:"+ msg);
	}
	@RabbitListener(queues = "direct.queue2")
	public void directListener2(String msg) {
		System.err.println("消费者二接收到 direct.queue2 的消息:"+ msg);
}

测试

Topic 交换机消息模型


Topic类型的ExchangeDirect相比,都是可以根据RoutingKey把消息路由到不同的队列。
只不过Topic类型Exchange可以让队列在绑定BindingKey 的时候使用通配符!- #:匹配一个或多个词 -*:匹配不多不少恰好 1 个词

举例:

  • item.#:能够匹配item.spu.insert 或者 item.spu
  • item.*:只能匹配item.spu

新建交换机,队列,绑定路由 key

publisher 测试类发送消息

@SpringBootTest
public class BaseTest {
	@Autowired
	private RabbitTemplate rabbitTemplate;
	@Test
	public void testSendTopicMessage() {
		String exchangeName = "dudu.topic";
		String message = "测试Topic消息模型";
		rabbitTemplate.convertAndSend(exchangeName,"china.message",message+"中国消息");
		rabbitTemplate.convertAndSend(exchangeName,"blue.news",message+"蓝色新闻");
		rabbitTemplate.convertAndSend(exchangeName,"yellow.news",message+"新闻");
	}
}

consumer 配置监听消息

@Component
public class RabbitMQListener {
	// 监听 topic 交换机消息模型 work.queue队列
	@RabbitListener(queues = "topic.queue1")
	public void topictListener1(String msg) {
		System.out.println("消费者一接收到 topic.queue1 的消息:"+ msg);
	}
	@RabbitListener(queues = "topic.queue2")
	public void topicListener2(String msg) {
		System.err.println("消费者二接收到 topic.queue2 的消息:"+ msg);
	}
}

测试

声明队列和交换机

若 mq 没有以方法名的交换机或队列, 则根据方法中 return 的新建交换机和队列

DirectConfig

@Configuration
public class DirectConfig {
	/**
	 * 声明交换机		若mq没有名为 fanoutExchange 的交换机, 则创建名为 hmall.direct 的交换机
	 * @return Direct类型交换机
	 */
	@Bean
	public DirectExchange directExchange(){
		return ExchangeBuilder.directExchange("hmall.direct").build();
	}

	/**
	 * 第1个队列
	 */
	@Bean
	public Queue directQueue1(){
		return new Queue("direct.queue1");
	}

	/**
	 * 绑定队列和交换机
	 */
	@Bean
	public Binding bindingQueue1WithRed(Queue directQueue1, DirectExchange directExchange){
		return BindingBuilder.bind(directQueue1).to(directExchange).with("red");
	}
	/**
	 * 绑定队列和交换机
	 */
	@Bean
	public Binding bindingQueue1WithBlue(Queue directQueue1, DirectExchange directExchange){
		return BindingBuilder.bind(directQueue1).to(directExchange).with("blue");
	}

	/**
	 * 第2个队列
	 */
	@Bean
	public Queue directQueue2(){
		return new Queue("direct.queue2");
	}

	/**
	 * 绑定队列和交换机
	 */
	@Bean
	public Binding bindingQueue2WithRed(Queue directQueue2, DirectExchange directExchange){
		return BindingBuilder.bind(directQueue2).to(directExchange).with("red");
	}
	/**
	 * 绑定队列和交换机
	 */
	@Bean
	public Binding bindingQueue2WithYellow(Queue directQueue2, DirectExchange directExchange){
		return BindingBuilder.bind(directQueue2).to(directExchange).with("yellow");
	}

FanoutConfig

@Configuration
public class FanoutConfig {
	/**
	 * 声明交换机 若mq没有名为 fanoutExchange 的交换机,则创建名为 dddddddddddddddddddddddddddddddddddddddddddddddddddddd.fanout 的交换机
	 * @return Fanout类型交换机
	 */
	@Bean
	public FanoutExchange fanoutExchange(){
		//ExchangeBuilder.fanoutExchange("").build();
		return new FanoutExchange("dddddddddddddddddddddddddddddddddddddddddddddddddddddd.fanout");
	}
	/**
	 * 第1个队列
	 */
	@Bean
	public Queue fanoutQueue1(){
		//QueueBuilder.durable("").build();
		return new Queue("fanoutdddddddddddddddddddddddddddddddddddddddddddddddddddddd.queue1");
	}
	/**
	 * 绑定队列和交换机
	 */
	@Bean
	public Binding bindingQueue1(Queue fanoutQueue1, FanoutExchange fanoutExchange){
		return BindingBuilder.bind(fanoutQueue1).to(fanoutExchange);
	}
	/**
	 * 第2个队列
	 */
	@Bean
	public Queue fanoutQueue2(){
		return new Queue("fanoutdddddddddddddddddddddddddddddddddddddddddddddddddddddd.queue2");
	}
	/**
	 * 绑定队列和交换机
	 */
	@Bean
	public Binding bindingQueue2(Queue fanoutQueue2, FanoutExchange fanoutExchange){
		return BindingBuilder.bind(fanoutQueue2).to(fanoutExchange);
	}

声明队列和交换机(注解)

AnnotationDirect

@Configuration
public class AnnotationDirect {
	@RabbitListener(bindings = @QueueBinding(
			value = @Queue(name = "direct.queue1"),
			exchange = @Exchange(name = "dudu.direct", type = ExchangeTypes.DIRECT),
			key = {"red", "blue"}
	))
	public void listenDirectQueue1(String msg){
		System.out.println("消费者1接收到direct.queue1的消息:【" + msg + "】");
	}

	@RabbitListener(bindings = @QueueBinding(
			value = @Queue(name = "direct.queue2"),
			exchange = @Exchange(name = "dudu.direct", type = ExchangeTypes.DIRECT),
			key = {"red", "yellow"}
	))
	public void listenDirectQueue2(String msg){
		System.out.println("消费者2接收到direct.queue2的消息:【" + msg + "】");
	}
}

AnnotationTopic

@Configuration
public class AnnotationTopic {
	@RabbitListener(bindings = @QueueBinding(
			value = @Queue(name = "topic.queue1"),
			exchange = @Exchange(name = "dudu.topic", type = ExchangeTypes.TOPIC),
			key = "china.#"
	))
	public void listenTopicQueue1(String msg){
		System.out.println("消费者1接收到topic.queue1的消息:【" + msg + "】");
	}

	@RabbitListener(bindings = @QueueBinding(
			value = @Queue(name = "topic.queue2"),
			exchange = @Exchange(name = "dudu.topic", type = ExchangeTypes.TOPIC),
			key = "#.news"
	))
	public void listenTopicQueue2(String msg){
		System.out.println("消费者2接收到topic.queue2的消息:【" + msg + "】");
	}
}

消息转换器

默认情况下 Spring 采用的序列化方式是 JDK 序列化,JDK 序列化存在下列问题:数据体积过大、有安全漏洞、可读性差

publisher 测试类 发一个 map 消息

@SpringBootTest
public class BaseTest {
	@Autowired
	private RabbitTemplate rabbitTemplate;
	@Test
	public void testSendMessageMap() {
		Map<String, Object> map = new HashMap<>();
		map.put("name", "张三");
		map.put("age", 18);
		rabbitTemplate.convertAndSend("object.queue",map);
	}
}

队列就手动在 mq 创建一个 object.queue
这时候消息监听服务开着的话就会报错


在 mq 上查看发送的消息

配置 JSON 转换器

publisherconsumer两个服务中都引入依赖:

<dependency>
    <groupId>com.fasterxml.jackson.dataformat</groupId>
    <artifactId>jackson-dataformat-xml</artifactId>
</dependency>

publisherconsumer两个服务的启动类中添加一个 Bean 即可或者写一个配置类把 bean 注入

@Configuration
public class MessageConverterConfig {
	@Bean
	public MessageConverter messageConverter(){
		// 1.定义消息转换器
		Jackson2JsonMessageConverter jackson2JsonMessageConverter = new Jackson2JsonMessageConverter();
		// 2.配置自动创建消息id,用于识别不同消息,也可以在业务中基于ID判断是否是重复消息
		jackson2JsonMessageConverter.setCreateMessageIds(true);
		return jackson2JsonMessageConverter;
	}
}

consumer 监听 object.queue

@Component
public class RabbitMQListener {
	// 监听 object.queue 队列
	@RabbitListener(queues = "object.queue")
	public void listenSimpleQueueMessage(Map<String, Object> msg) throws InterruptedException {
		System.out.println("消费者接收到object.queue消息:【" + msg + "】");
	}
}

测试

发送者的可靠性

修改 publisher 配置问价

spring:
  rabbitmq:
    host: 192.168.64.100 # 你的虚拟机IP
    port: 5672 # 端口
    virtual-host: /dudu # 虚拟主机
    username: dudu # 用户名
    password: 123456 # 密码
    # 生产者重试机制
    connection-timeout: 1s #设置MQ的连接超时时间
    template:
      retry:
        enabled: true # 开启超时重试机制  SpringAMQP提供的重试机制是阻塞式的重试,也就是说多次重试等待的过程中,当前线程是被阻塞的。
        # 如果对于业务性能有要求,建议禁用重试机制。如果一定要使用,请合理配置等待时长和重试次数,当然也可以考虑使用异步线程来执行发送消息的代码。
        initial-interval: 1000ms # 失败后的初始等待时间
        multiplier: 1 # 失败后下次的等待时长倍数,下次等待时长 = initial-interval * multiplier
        max-attempts: 3 # 最大重试次数
    # 生产者确认机制
    # publisher-confirm-type`有三种模式可选:-
    # `none`:关闭confirm机制,simple`:同步阻塞等待MQ的回执,correlated`:MQ异步回调返回回执
    #一般我们推荐使用`correlated`,回调机制。
    publisher-confirm-type: correlated # 开启publisher confirm机制,并设置confirm类型
    publisher-returns: true # 开启publisher return机制

定义 ReturnCallback

每个RabbitTemplate只能配置一个ReturnCallback,因此我们可以在配置类中统一设置。我们在 publisher 模块定义一个配置类:MqConfig

@AllArgsConstructor
@Configuration
public class MqConfig {
	private final RabbitTemplate rabbitTemplate;

	@PostConstruct
	public void init(){
		rabbitTemplate.setReturnsCallback(new RabbitTemplate.ReturnsCallback() {
			@Override
			public void returnedMessage(@Nonnull ReturnedMessage returnedMessage) {
				System.out.println("收到ReturnsCallback===========================");
				System.out.println("消息未进入队列"+returnedMessage.getMessage());
				System.out.println("交换机:"+returnedMessage.getExchange());
				System.out.println("路由键:"+returnedMessage.getRoutingKey());
				System.out.println("replyCode:"+returnedMessage.getReplyCode());
				System.out.println("replyText:"+returnedMessage.getReplyText());
			}
		});
		rabbitTemplate.setConfirmCallback((correlationData, ack, cause) -> {
			System.out.println("收到ConfirmCallback===========================");
			System.out.println("是否到交换机:"+correlationData);
			System.out.println("ack:"+ack);
			System.out.println("原因:"+cause);
			if (!ack){
				System.out.println("消息发送失败"+cause);
			}
		});
	}
}

也可以这样写

@Configuration
@AllArgsConstructor
public class MqConfig {
	@Bean
	public RabbitTemplate rabbitTemplate(CachingConnectionFactory factory) {
		// 开启消息进入Broker确认
		factory.setPublisherConfirmType(CachingConnectionFactory.ConfirmType.CORRELATED);
		// 开启消息未进入队列确认
		factory.setPublisherReturns(true);

		RabbitTemplate rabbitTemplate = new RabbitTemplate(factory);
		// 进入Broker时触发回调
		rabbitTemplate.setConfirmCallback((correlationData, b, s) -> {
			System.out.println("是否到交换机:"+correlationData);
			System.out.println("ack:"+b);
			System.out.println("原因:"+s);
			if (b) {
				System.out.println("消息进入Broker成功");
			} else {
				System.out.println("消息进入Broker失败");
			}
		});

		// Mandatory:为true时,消息通过交换器无法匹配到队列会返回给生产者 并触发MessageReturn,为false时,匹配不到会直接被丢弃
		rabbitTemplate.setMandatory(true);
		// 消息未进入队列时触发回调
		rabbitTemplate.setReturnsCallback(returnedMessage -> {
			System.out.println("消息未进入队列"+returnedMessage.getMessage());
			System.out.println("交换机:"+returnedMessage.getExchange());
			System.out.println("路由键:"+returnedMessage.getRoutingKey());
			System.out.println("replyCode:"+returnedMessage.getReplyCode());
			System.out.println("replyText:"+returnedMessage.getReplyText());
		});
		return rabbitTemplate;
	}
}

新建测试、并且添加 ConfirmCallback

@Test
void testPublisherConfirm() {
    // 1.创建CorrelationData
    CorrelationData cd = new CorrelationData();
    // 2.给Future添加ConfirmCallback
    cd.getFuture().addCallback(new ListenableFutureCallback<CorrelationData.Confirm>() {
        @Override
        public void onFailure(Throwable ex) {
            // 2.1.Future发生异常时的处理逻辑,基本不会触发
            log.error("send message fail", ex);
        }
        @Override
        public void onSuccess(CorrelationData.Confirm result) {
            // 2.2.Future接收到回执的处理逻辑,参数中的result就是回执内容
            if(result.isAck()){ // result.isAck(),boolean类型,true代表ack回执,false 代表 nack回执
                log.debug("发送消息成功,收到 ack!");
            }else{ // result.getReason(),String类型,返回nack时的异常描述
                log.error("发送消息失败,收到 nack, reason : {}", result.getReason());
            }
        }
    });
    // 3.发送消息
    rabbitTemplate.convertAndSend("hmall.direct", "q", "hello", cd);
}

新的版本好像没有这个 addCallback()方法了下面这个倒是可能也可以吧

@Test
	void testPublisherConfirm() throws InterruptedException {
		// 1.创建CorrelationData
		CorrelationData cd = new CorrelationData();
		cd.getFuture().whenComplete((confirm, throwable) -> {
			System.out.println("confirm: " + confirm + " throwable: " + throwable);
			if (confirm.isAck()) {
				System.out.println("消息发送成功,收到ack"+confirm.getReason());
			}else {
				System.out.println("消息发送失败,收到nack"+confirm.getReason());
			}
		});
		rabbitTemplate.convertAndSend("hmall.11direct", "blu1e", "hello",cd);
		Thread.sleep(2000);
	}

测试

总结

开启生产者确认比较消耗 MQ 性能,一般不建议开启。而且大家思考一下触发确认的几种情况:

  • 路由失败:一般是因为 RoutingKey 错误导致,往往是编程导致
  • 交换机名称错误:同样是编程错误导致
  • MQ 内部故障:这种需要处理,但概率往往较低。因此只有对消息可靠性要求非常高的业务才需要开启,而且仅仅需要开启 ConfirmCallback 处理 nack 就可以了。

MQ 的可靠性

说明:在开启持久化机制以后,如果同时还开启了生产者确认,那么 MQ 会在消息持久化以后才发送 ACK 回执,进一步确保消息的可靠性。

不过出于性能考虑,为了减少 IO 次数,发送到 MQ 的消息并不是逐条持久化到数据库的,而是每隔一段时间批量持久化。一般间隔在 100 毫秒左右,这就会导致 ACK 有一定的延迟,因此建议生产者确认全部采用异步方式。

交换机持久化

队列持久化

消息持久化

在控制台发消息时可以指定参数

代码实现

	@Test
	public void testSendMessage(){
		//队列名称
		String queueName = "base.queue";
		//消息
		String message = "基本消息模型测试";
		//发送消息
		//设置消息持久化
		rabbitTemplate.setMandatory(true);
		rabbitTemplate.convertAndSend(queueName,message);
	}

LazyQueue

在 3.12 版本之后,LazyQueue 已经成为所有队列的默认格式。因此官方推荐升级 MQ 为 3.12 版本或者所有队列都设置为 LazyQueue 模式。
在添加队列的时候,添加x-queue-mod=lazy参数即可设置队列为 Lazy 模式:

代码

@Bean
public Queue lazyQueue(){
    return QueueBuilder
            .durable("lazy.queue")
            .lazy() // 开启Lazy模式
            .build();
}

注解方式

@RabbitListener(queuesToDeclare = @Queue(
        name = "lazy.queue",
        durable = "true",
        arguments = @Argument(name = "x-queue-mode", value = "lazy")
))
public void listenLazyQueue(String msg){
    log.info("接收到 lazy.queue的消息:{}", msg);
}

更新已有队列为 lazy 模式

命令

rabbitmqctl set_policy Lazy "^lazy-queue$" '{"queue-mode":"lazy"}' --apply-to queues

命令解读:

  • rabbitmqctl :RabbitMQ 的命令行工具
  • set_policy :添加一个策略
  • Lazy :策略名称,可以自定义
  • "^lazy-queue$" :用正则表达式匹配队列的名字
  • '{"queue-mode":"lazy"}' :设置队列模式为 lazy 模式
  • --apply-to queues:策略的作用对象,是所有的队列

消费者的可靠性

消费者确认机制

当消费者处理消息结束后,应该向 RabbitMQ 发送一个回执,告知 RabbitMQ 自己消息处理状态。回执有三种可选值:

  • ack:成功处理消息,RabbitMQ 从队列中删除该消息
  • nack:消息处理失败,RabbitMQ 需要再次投递消息
  • reject:消息处理失败并拒绝该消息,RabbitMQ 从队列中删除该消息

一般 reject 方式用的较少,除非是消息格式有问题,那就是开发问题了。因此大多数情况下我们需要将消息处理的代码通过try catch机制捕获,消息处理成功时返回 ack,处理失败时返回 nack.

由于消息回执的处理代码比较统一,因此 SpringAMQP 帮我们实现了消息确认。并允许我们通过配置文件设置 ACK 处理方式,有三种模式:

  • none:不处理。即消息投递给消费者后立刻 ack,消息会立刻从 MQ 删除。非常不安全,不建议使用

  • manual:手动模式。需要自己在业务代码中调用 api,发送ackreject,存在业务入侵,但更灵活

  • auto:自动模式。SpringAMQP 利用 AOP 对我们的消息处理逻辑做了环绕增强,当业务正常执行时则自动返回ack. 当业务出现异常时,根据异常判断返回不同结果:

    • 如果是业务异常,会自动返回nack
    • 如果是消息处理或校验异常,自动返回reject;

修改 consumer 的 yml 文件

spring:
  rabbitmq:
    host: 192.168.64.100 # 你的虚拟机IP
    port: 5672 # 端口
    virtual-host: /dudu # 虚拟主机
    username: dudu # 用户名
    password: 123456 # 密码
    listener:
      simple:
        prefetch: 1 # 每次只能获取一条消息,处理完成才能获取下一个消息
        # 确认模式
        # none:不处理。即消息投递给消费者后立刻ack,消息会立刻从MQ删除。非常不安全,不建议使用
        # manual:手动确认。即消费者处理完消息后,需要手动ack。
        # auto:自动确认。即消费者处理完消息后,自动ack,消息会从MQ删除。如果是业务异常,会自动返回`nack` 消息处理或校验异常,自动返回`reject`消息不会从MQ删除
        acknowledge-mode: auto

失败重试机制

修改 consumer 的配置

spring:
  rabbitmq:
    host: 192.168.64.100 # 你的虚拟机IP
    port: 5672 # 端口
    virtual-host: /dudu # 虚拟主机
    username: dudu # 用户名
    password: 123456 # 密码
    listener:
      simple:
        prefetch: 1 # 每次只能获取一条消息,处理完成才能获取下一个消息
        # 确认模式
        # none:不处理。即消息投递给消费者后立刻ack,消息会立刻从MQ删除。非常不安全,不建议使用
        # manual:手动确认。即消费者处理完消息后,需要手动ack,消息不会从MQ删除。
        # auto:自动确认。即消费者处理完消息后,自动ack,消息会从MQ删除。如果是业务异常,会自动返回`nack` 消息处理或校验异常,自动返回`reject`
        acknowledge-mode: auto
        # 失败重试机制
        retry:
          enabled: true # 开启消费者失败重试
          initial-interval: 1000ms # 初识的失败等待时长为1秒
          multiplier: 1 # 失败的等待时长倍数,下次等待时长 = multiplier * last-interval
          max-attempts: 3 # 最大重试次数
          stateless: true # true无状态;false有状态。如果业务中包含事务,这里改为false
  • 消费者在失败后消息没有重新回到 MQ 无限重新投递,而是在本地重试了 3 次
  • 本地重试 3 次以后,抛出了AmqpRejectAndDontRequeueException异常。查看 RabbitMQ 控制台,发现消息被删除了,说明最后 SpringAMQP 返回的是reject

结论:

  • 开启本地重试时,消息处理过程中抛出异常,不会 requeue 到队列,而是在消费者本地重试
  • 重试达到最大次数后,Spring 会返回 reject,消息会被丢弃

失败处理策略

本地测试达到最大重试次数后,消息会被丢弃。这在某些对于消息可靠性要求较高的业务场景下,显然不太合适了。
因此 Spring 允许我们自定义重试次数耗尽后的消息处理策略,这个策略是由MessageRecovery接口来定义的,它有 3 个不同实现:

  • RejectAndDontRequeueRecoverer:重试耗尽后,直接reject,丢弃消息。默认就是这种方式
  • ImmediateRequeueMessageRecoverer:重试耗尽后,返回nack,消息重新入队
  • RepublishMessageRecoverer:重试耗尽后,将失败消息投递到指定的交换机

比较优雅的一种处理方案是RepublishMessageRecoverer,失败后将消息投递到一个指定的,专门存放异常消息的队列,后续由人工集中处理。

ErrorMessageConfig 配置类

@Configuration
// 开启重试机制 这个配置类才会生效
@ConditionalOnProperty(name = "spring.rabbitmq.listener.simple.retry.enabled", havingValue = "true")
public class ErrorMessageConfig {
	@Bean
	public DirectExchange errorMessageExchange(){
		return new DirectExchange("error.direct");
	}
	@Bean
	public Queue errorQueue(){
		return new Queue("error.queue", true);
	}
	@Bean
	public Binding errorBinding(Queue errorQueue, DirectExchange errorMessageExchange){
		return BindingBuilder.bind(errorQueue).to(errorMessageExchange).with("error");
	}

	@Bean
	public MessageRecoverer republishMessageRecoverer(RabbitTemplate rabbitTemplate){
		return new RepublishMessageRecoverer(rabbitTemplate, "error.direct", "error");
	}
}

延迟消息

死信交换机

当一个队列中的消息满足下列情况之一时,可以成为死信(dead letter):

  • 消费者使用basic.rejectbasic.nack声明消费失败,并且消息的requeue参数设置为 false
  • 消息是一个过期消息,超时无人消费
  • 要投递的队列消息满了,无法投递
    如果一个队列中的消息已经成为死信,并且这个队列通过**dead-letter-exchange**属性指定了一个交换机,那么队列中的死信就会投递到这个交换机中,而这个交换机就称为死信交换机(Dead Letter Exchange)。而此时加入有队列与死信交换机绑定,则最终死信就会被投递到这个队列中。

死信交换机有什么作用呢?

  1. 收集那些因处理失败而被拒绝的消息
  2. 收集那些因队列满了而被拒绝的消息
  3. 收集因 TTL(有效期)到期的消息

注意:
RabbitMQ 的消息过期是基于追溯方式来实现的,也就是说当一个消息的 TTL 到期以后不一定会被移除或投递到死信交换机,而是在消息恰好处于队首时才会被处理。
当队列中消息堆积很多的时候,过期消息可能不会被按时处理,因此你设置的 TTL 时间不一定准确。

DelayExchange 插件

安装
基于 Docker 安装,所以需要先查看 RabbitMQ 的插件目录对应的数据卷

docker volume inspect mq-plugins

结果如下

[
    {
        "CreatedAt": "2024-06-19T09:22:59+08:00",
        "Driver": "local",
        "Labels": null,
        "Mountpoint": "/var/lib/docker/volumes/mq-plugins/_data",
        "Name": "mq-plugins",
        "Options": null,
        "Scope": "local"
    }
]

插件目录被挂载到了/var/lib/docker/volumes/mq-plugins/_data这个目录,我们上传插件到该目录下。

接下来执行命令,安装插件:

docker exec -it mq rabbitmq-plugins enable rabbitmq_delayed_message_exchange

声明延迟交换机

基于注解方式:

@RabbitListener(bindings = @QueueBinding(
        value = @Queue(name = "delay.queue", durable = "true"),
        exchange = @Exchange(name = "delay.direct", delayed = "true"),
        key = "delay"
))
public void listenDelayMessage(String msg){
    log.info("接收到delay.queue的延迟消息:{}", msg);
}

基于@Bean的方式:

import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.core.*;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class DelayExchangeConfig {
    @Bean
    public DirectExchange delayExchange(){
        return ExchangeBuilder
                .directExchange("delay.direct") // 指定交换机类型和名称
                .delayed() // 设置delay的属性为true
                .durable(true) // 持久化
                .build();
    }
    @Bean
    public Queue delayedQueue(){
        return new Queue("delay.queue");
    }
    @Bean
    public Binding delayQueueBinding(){
        return BindingBuilder.bind(delayedQueue()).to(delayExchange()).with("delay");
    }
}

发送延迟消息

@SpringBootTest
public class BaseTest {
	@Autowired
	private RabbitTemplate rabbitTemplate;
	@Test
	void testPublisherDelayMessage() {
		// 1.创建消息
		String message = "hello, delayed message";
		// 2.发送消息,利用消息后置处理器添加消息头
		rabbitTemplate.convertAndSend("delay.direct", "delay", message, new MessagePostProcessor() {
			@Override
			public Message postProcessMessage(Message message) throws AmqpException {
				// 添加延迟消息属性  这里的.setHeader("x-delay", 10000)替代了setDelay(10000)
				message.getMessageProperties().setHeader("x-delay", 10000);
				return message;
			}
		});
		System.out.println("消息发送成功"+ LocalDateTime.now());
	}
}

消息发送十秒后,消费者接收到消息

注意: 延迟消息插件内部会维护一个本地数据库表,同时使用 Elang Timers 功能实现计时。如果消息的延迟时间设置较长,可能会导致堆积的延迟消息非常多,会带来较大的 CPU 开销,同时延迟消息的时间会存在误差。
因此,不建议设置延迟时间过长的延迟消息

假如订单超时支付时间为 30 分钟,理论上说我们应该在下单时发送一条延迟消息,延迟时间为 30 分钟。这样就可以在接收到消息时检验订单支付状态,关闭未支付订单。
但是大多数情况下用户支付都会在 1 分钟内完成,我们发送的消息却要在 MQ 中停留 30 分钟,额外消耗了 MQ 的资源。因此,我们最好多检测几次订单支付状态,而不是在最后第 30 分钟才检测。
例如:我们在用户下单后的第 10 秒、20 秒、30 秒、45 秒、60 秒、1 分 30 秒、2 分、…30 分分别设置延迟消息,如果提前发现订单已经支付,则后续的检测取消即可。
这样就可以有效避免对 MQ 资源的浪费了。

整体视图

  • 18
    点赞
  • 30
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值