1. 死信机制
1.1 死信(dead letter)
当一个队列中的消息满足以下情况之一,则被称为死信
- 消费者使用basic.reject或者basic.nack生命消费失败,并且消息的requeue参数设置为false
- 消息是一个过期消息,超时没人消费
- 要投递的队列消息满了,无法投递
1.2 死信队列
- 就是一个普通的队列,用来存放死信的
- 只不过时设置了一些参数
1.3 示例
- simple.queue:就是一个普通的队列,但是这个队列上绑定了 一些参数。指明了死信要转投到哪个交换机
// 声明普通的 simple.queue队列,并且为其指定死信交换机:dl.direct
@Bean
public Queue simpleQueue2(){
return QueueBuilder.durable("simple.queue") // 指定队列名称,并持久化
.deadLetterExchange("dl.direct") // 指定死信交换机
.build();
}
// 声明死信交换机 dl.direct
@Bean
public DirectExchange dlExchange(){
return new DirectExchange("dl.direct", true, false);
}
// 声明存储死信的队列 dl.queue
@Bean
public Queue dlQueue(){
return new Queue("dl.queue", true);
}
// 将死信队列 与 死信交换机绑定
@Bean
public Binding dlBinding(){
return BindingBuilder.bind(dlQueue()).to(dlExchange()).with("simple");
}
2. 惰性队列
2.1 消息堆积
- 生产者发送消息的速度远远超过消费者处理消息的速度,导致队列满,早期的消息成为死信,可能会被丢弃
2.2 消息堆积的解决方法
- 增加更多的消费者,提高消费者的速度
- 消费者内开启线程池,加快消费处理速度
- 扩大队列容量,提高堆积上限
2.3 惰性队列
- 接受消息后直接存入磁盘而非内存
- 消费者要消费信息时才会从磁盘中读取并加载到内存
- 支持数百万条的数据存储
2.4 如何声明一个惰性队列
基于@Bean声明惰性队列
@Bean
public Queue lazyQueue() {
return QueueBuilder.durable("lazy.queue")
.lazy()
.build();
}
基于@RabbitListener声明lazyqueue
@RabbitListener(queuesToDeclare = @Queue(
name = "lazy.queue",
durable = "true",
arguments = @Argument(
name = "x-queue-mode",
value = "lazy"
)
))
public void listenLazyQueue(String msg) {
System.out.println("处理惰性队列中的消息" + msg);
}