MQ高级特性
1.削峰
设置 消费者
data:image/s3,"s3://crabby-images/b3bbf/b3bbffec97236b70b6c168275b95d677b91d503b" alt=""
测试 添加多条消息
data:image/s3,"s3://crabby-images/39e9a/39e9af660ce980c06696a2e9439b0cf8acb869e7" alt=""
data:image/s3,"s3://crabby-images/96a78/96a78f955d0a618fc7561169ccd16c0e50dff2ae" alt=""
拉取消息 每隔20秒拉取一次 一次拉取五条 然后在20秒内一条一条消费
data:image/s3,"s3://crabby-images/ab246/ab246ae906cb79f119fdac6802e9014f2ef8040d" alt=""
data:image/s3,"s3://crabby-images/051ce/051ceed4a3efe234915699672a064c729a749c4a" alt=""
data:image/s3,"s3://crabby-images/c807d/c807daf37d5f03c0ab72fb22794c75644ac7111b" alt=""
TTL
Time To Live(存活时间/过期时间)。
当消息到达存活时间后,还没有被消费,会被自动清除。
RabbitMQ可以对消息设置过期时间,也可以对整个队列(Queue)设置过期时间。
可以在管理台新建队列、交换机,绑定
1.图形化操作
添加队列
data:image/s3,"s3://crabby-images/87dfa/87dfacba6e21696d58b6a5364c9f24c0fd3bf8f0" alt=""
添加交换机
data:image/s3,"s3://crabby-images/5588d/5588d50b9df888107e6f80d28daea9356dc1b82c" alt=""
将交换机和对应的队列进行绑定
data:image/s3,"s3://crabby-images/dae1e/dae1e97c5a59c4248a9e97efc9c641e8f70c4840" alt=""
data:image/s3,"s3://crabby-images/dbc03/dbc033905dd0c04ba36b462587a84f4bc02cf860" alt=""
时间结束 , 消息失效
data:image/s3,"s3://crabby-images/32260/3226071dd4e4135948747689ed7df5091b26d3b4" alt=""
2.代码实现
配置 生产者
data:image/s3,"s3://crabby-images/e72be/e72be3400ef7671f09e1be80edbd165dee8b522d" alt=""
@Configuration
public class TopicMqTtlConfig {
@Value("${mq.exchange.name}")
private String EXCHANGENAME;
@Value("${mq.queue.name1}")
private String QUEUENAME1;
@Value("${mq.queue.name2}")
private String QUEUENAME2;
// 1
// . 交换机
@Bean("ex1")
public Exchange getExchange(){
Exchange exchange = ExchangeBuilder.topicExchange(EXCHANGENAME).durable(false).build();
return exchange;
}
// 2。 队列
@Bean("queue1")
public Queue getQueue1(){
Queue queue = QueueBuilder.nonDurable(QUEUENAME1)
.withArgument("x-message-ttl",30000)//过期时间30秒
.withArgument("x-max-length",10)//队列中最多接收10条消息超过10条的部分废弃
.build();
return queue;
}
@Bean("queue2")
public Queue getQueue2(){
Queue queue2 = QueueBuilder.nonDurable(QUEUENAME2)
.withArgument("x-message-ttl",300000000)//过期时间30秒
.build();
return queue2;
}
// 3. 交换机和队列进行绑定
@Bean("binding1")
public Binding bindQueue1ToExchange(@Qualifier("ex1") Exchange exchange,@Qualifier("queue1") Queue queue){
Binding binding1 = BindingBuilder.bind(queue).to(exchange).with("ttl1.*").noargs();
return binding1;
}
@Bean("binding2")
public Binding bindQueue2ToExchange(@Qualifier("ex1") Exchange exchange,@Qualifier("queue2") Queue queue){
Binding binding2 = BindingBuilder.bind(queue).to(exchange).with("ttl2.#").noargs();
return binding2;
}
}
data:image/s3,"s3://crabby-images/6496f/6496f696d0805039f70cbb7e24531c2650ca0782" alt=""
测试
data:image/s3,"s3://crabby-images/dbb18/dbb1862558fa27c30d2343d01cbbe8aac9b117d0" alt=""
添加成功 ttl1只接收10条
data:image/s3,"s3://crabby-images/caaf2/caaf2c204ba615da333daf48dca4e3804e9c8be1" alt=""
时间过期
data:image/s3,"s3://crabby-images/c5fb5/c5fb5878f0a25cc4ebdaae67d99093b78307c6aa" alt=""
死信队列
死信队列,英文缩写:DLX 。Dead Letter Exchange(死信交换机,因为其他MQ产品中没有交换机的概念),当消息成为Dead message后,可以被重新发送到另一个交换机,这个交换机就是DLX。
比如消息队列的消息过期,如果绑定了死信交换器,那么该消息将发送给死信交换机
data:image/s3,"s3://crabby-images/f2f50/f2f50caf3b9e177cdf4afb15794d2ea1cc10e33f" alt=""
消息在什么情况下会成为死信?(面试会问)
1.队列消息长度到最大的限制
最大的长度设置为10当第11条消息进来的时候就会成为死信
- 消费者拒接消费消息,basicNack/basicReject,并且不把消息重新放入原目标队列,requeue=false(不重新回到队列中)
设置消费者为手动签收的状态
- 原队列存在消息过期设置,消息到达超时时间未被消费;
队列绑定交换机的方式是什么?
给队列设置参数: x-dead-letter-exchange 和 x-dead-letter-routing-key
data:image/s3,"s3://crabby-images/f2f50/f2f50caf3b9e177cdf4afb15794d2ea1cc10e33f" alt=""
// 1. 交换机 :正常的交换机 死信交换机
// 2.队列 :正常的 死信
//3.绑定 正常ex - 正常的que
正常的que和死信交换机
死信ex-死信queue
data:image/s3,"s3://crabby-images/da016/da016f937245b0edd80814e9105479f62cc22526" alt=""
2.代码实现
@Configuration
public class TopicMqDeadConfig {
@Value("${mq1.exchange.name1}")
private String EXCHANGENAME;
@Value("${mq1.exchange.name2}")
private String DEADEXCHANGE;
@Value("${mq1.queue.name1}")
private String QUEUENAME1;
@Value("${mq1.queue.name2}")
private String QUEUENAME2;
// 声明正常交换机
@Bean("ex1")
public Exchange getExchange(){
Exchange exchange = ExchangeBuilder.topicExchange(EXCHANGENAME).durable(false).build();
return exchange;
}
// 正常队列
@Bean("queue1")
public Queue getQueue1(){
Queue queue = QueueBuilder.nonDurable(QUEUENAME1)
.withArgument("x-message-ttl",30000)//过期时间30秒
.withArgument("x-dead-letter-exchange",DEADEXCHANGE)
.withArgument("x-dead-letter-routing-key","dead.test")//将正常队列与死信交换机,死信队列绑定
//.withArgument("x-max-length",10)//队列中最多接收10条消息超过10条的部分废弃
.build();
return queue;
}
// 交换机和队列进行绑定
@Bean("binding1")
public Binding bindQueue1ToExchange(@Qualifier("ex1") Exchange exchange,@Qualifier("queue1") Queue queue){
Binding binding1 = BindingBuilder.bind(queue).to(exchange).with("normal.*").noargs();
return binding1;
}
// 声明死信交换机
@Bean("ex2")
public Exchange getDeadExchange(){
Exchange exchange = ExchangeBuilder.topicExchange(DEADEXCHANGE).durable(false).build();
return exchange;
}
//死信队列
@Bean("queue2")
public Queue getQueue2(){
Queue queue2 = QueueBuilder.nonDurable(QUEUENAME2)
.build();
return queue2;
}
// 死信交换机和死信队列进行绑定
@Bean("binding2")
public Binding bindQueue2ToExchange(@Qualifier("ex2") Exchange exchange,@Qualifier("queue2") Queue queue){
Binding binding2 = BindingBuilder.bind(queue).to(exchange).with("dead.*").noargs();
return binding2;
}
}
data:image/s3,"s3://crabby-images/ea1a9/ea1a92571a973bad163aea63ae5834f9785d715d" alt=""
测试
data:image/s3,"s3://crabby-images/b1545/b154570dd4f49d58720b2017e6995eb9b2c4ad53" alt=""
data:image/s3,"s3://crabby-images/0ac2f/0ac2fd2de9828d6deea1069b4c3ff22b8c0982a5" alt=""
如果程序出现错误 拒绝签收
监听正常队列
data:image/s3,"s3://crabby-images/3a40f/3a40f48937f814eeaabea6953e18839b23b7cb0f" alt=""
发送消息 启动测试
data:image/s3,"s3://crabby-images/8d34e/8d34ee0acbd30fd6f2100b7cdb246c3c77f23979" alt=""
data:image/s3,"s3://crabby-images/457d8/457d85d97273a9c8e6b21083ef84281d3dc3e02d" alt=""
data:image/s3,"s3://crabby-images/25356/25356f5e7c9544af4388d7d095032c50145aa1ab" alt=""
总结:
-
死信交换机和死信队列和普通的没有区别
-
当消息成为死信后,如果该队列绑定了死信交换机,则消息会被死信交换机重新路由到死信队列
-
消息成为死信的三种情况:
-
队列消息长度到达限制;
-
消费者拒接消费消息,并且不重回队列;
-
原队列存在消息过期设置,消息到达超时时间未被消费;
延迟队列
延迟队列,即消息进入队列后不会立即被消费,只有到达指定时间后,才会被消费。
需求:
-
- 下单后,30分钟未支付,取消订单,回滚库存
-
- 新用户注册成功7天后,发送短信问候。
data:image/s3,"s3://crabby-images/8a1b0/8a1b0702a0f0374f33bf37b3b050939941cbaa1e" alt=""
实现方式:
-
定时器
-
死信队列
在RabbitMQ中并未提供延迟队列功能。但是可以使用:TTL+死信队列
组合实现延迟队列的效果。
data:image/s3,"s3://crabby-images/0f474/0f474df0ccc0053562f1212e471f6c0f1948214d" alt=""
1.配置
添加依赖
<!--2. rabbitmq-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!--nacos 配置中心-->
<!--配置中心-->
<dependency>
<groupId>com.alibaba.cloud</groupId>
<artifactId>spring-cloud-starter-alibaba-nacos-config</artifactId>
</dependency>
<!-- application bootstrap -->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-bootstrap</artifactId>
</dependency>
<!-- nacos-->
<dependency>
<groupId>com.alibaba.cloud</groupId>
<artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
</dependency>
<dependency>
<groupId>com.example</groupId>
<artifactId>sys-comm</artifactId>
<version>0.0.1-SNAPSHOT</version>
</dependency>
data:image/s3,"s3://crabby-images/ef6b1/ef6b16f358196af21a494431eeb4db9a36472bcf" alt=""
修改配置
data:image/s3,"s3://crabby-images/1fb28/1fb28ba6f43be557f089949146a9db35b496f4c0" alt=""
data:image/s3,"s3://crabby-images/46f37/46f37fa4646d656302cc64c4993cd9ec26768aac" alt=""
data:image/s3,"s3://crabby-images/d3461/d346167c9d97bcf72f20f62d1ffe9dd71d9423e5" alt=""
2.代码实现
data:image/s3,"s3://crabby-images/780b3/780b301bdb33c938da6e5e286f618fce600897f8" alt=""
创建实体类
data:image/s3,"s3://crabby-images/bddbf/bddbff5be33969341e63c67950cc4cd5b46233d1" alt=""
发送消息 测试
data:image/s3,"s3://crabby-images/49bd9/49bd9ee6b0d3bfe2708aabfb9de950ab674c9fb2" alt=""
data:image/s3,"s3://crabby-images/f9545/f9545919b05eef1ac83b47c4a3f4aa13fa8421eb" alt=""
过期后放入死信队列
data:image/s3,"s3://crabby-images/e4eec/e4eec1f9b218e8dff733812dd57ef2bcf8fefda1" alt=""
data:image/s3,"s3://crabby-images/24846/248466adf9a4510f575a5d6fa02aae2319a561e3" alt=""
添加依赖
<dependency>
<groupId>cn.hutool</groupId>
<artifactId>hutool-all</artifactId>
<version>5.8.16</version>
</dependency>
data:image/s3,"s3://crabby-images/86703/867031b7100ca5079fab9abd8a06d4866c25a971" alt=""
将json数据转化为对象
data:image/s3,"s3://crabby-images/4b331/4b331006604eb933d048afa0a88ca8b1200e14c0" alt=""
data:image/s3,"s3://crabby-images/9fa89/9fa89e4d674cd11d53277039de5a31c6def3219d" alt=""
获取成功
data:image/s3,"s3://crabby-images/361d1/361d16b5ffb959625b89dfb2c314dd756b675964" alt=""
3.连接数据库
创建表
data:image/s3,"s3://crabby-images/c98f6/c98f6baf56f84a55ec827746c8d08bf75b9c9dcd" alt=""
创建测试类
@RestController
@RequestMapping("order")
public class OrderController {
@Value("${mq1.exchange.name1}")
private String EXCHANGENAME;
//
@Resource
private RabbitTemplate rabbitTemplate;
@GetMapping
public Result aaa(TabOrder order){
//1. 消息 存放到mq里面
String s = JSONUtil.toJsonStr(order);
// openfeign -- 数据添加到数据库里面
rabbitTemplate.convertAndSend(EXCHANGENAME, "normal.test", s);
return Result.success(s);
}
}
data:image/s3,"s3://crabby-images/0b752/0b75247eb64d3f264d5ce7600bd3fda913f4d225" alt=""
监听normal
import javax.annotation.Resource;
@Component
public class XiaoFeng implements ChannelAwareMessageListener {
@Resource
private TabOrderMapper orderMapper;
@Override
@RabbitListener(queues = "test_queue_normal")
public void onMessage(Message message, Channel channel) throws Exception {
//Thread.sleep(2000);// 20s
byte[] body = message.getBody();
String s = new String(body);
System.out.println(s);
// 将字符串转化为 对象
long deliveryTag = message.getMessageProperties().getDeliveryTag();
try{
TabOrder order = JSONUtil.toBean(s, TabOrder.class);
// 将订单的信息 报讯到数据库里面
int insert = orderMapper.insert(order);
channel.basicAck(deliveryTag,true); //
}catch(Exception e){
//long deliveryTag, boolean multiple, boolean requeue
System.out.println("拒绝签收消息");
channel.basicNack(deliveryTag,true,false);// 死信消息
}
}
}
data:image/s3,"s3://crabby-images/a29c8/a29c8cbd1dc1f96b2035ffc3fc3901ea5588f996" alt=""
监听dead
@Component
public class YanChi implements ChannelAwareMessageListener {
@Resource
private TabOrderMapper orderMapper;
@Override
@RabbitListener(queues = "test_queue_dead")
public void onMessage(Message message, Channel channel) throws Exception {
//Thread.sleep(2000);// 20s
byte[] body = message.getBody();
String s = new String(body);
System.out.println(s);
// 将字符串转化为 对象
long deliveryTag = message.getMessageProperties().getDeliveryTag();
try{
TabOrder order = JSONUtil.toBean(s, TabOrder.class);
// order 的状态
TabOrder tabOrder = orderMapper.selectById(order.getId());
if(tabOrder.getStatus()==1){
// 取消
tabOrder.setStatus(3);
}
orderMapper.updateById(tabOrder);
channel.basicAck(deliveryTag,true); //
}catch(Exception e){
//long deliveryTag, boolean multiple, boolean requeue
System.out.println("拒绝签收消息");
channel.basicNack(deliveryTag,true,false);// 死信消息
}
}
}
data:image/s3,"s3://crabby-images/231ca/231caff4654a7b11ea87374e16e56d2cb356ea38" alt=""
测试
data:image/s3,"s3://crabby-images/56478/56478eb844ba2324db24a446b398270dd665d023" alt=""
成功
data:image/s3,"s3://crabby-images/db222/db222aa3ba509790b826a84f9a9d6a2ec6ff9aa6" alt=""