一、延迟队列设计
队列中的消息成为死信(Dead Letter)的几种情况:
1) The message is rejected (basic.reject or basic.nack) with requeue=false,
2) The TTL for the message expires; or
3) The queue length limit is exceeded.
而通过配置队列的x-dead-letter-exchange及x-dead-letter-routing-key键值,Dead Letter就会被重新发送到指定的DLX中。
如上图所示,注意延迟队列并没有消费者,并且设置参数:
x-dead-letter-exchange:DLX
x-dead-letter-routing-key:DLK
x-max-length:1000000 // length = 1million
x-message-ttl:30000 // TTL = 30s
生产者发布的消息路由到该延迟队列,30秒后消息成为死信并被重新发送到死信交换器DLX中;消费者订阅死信队列DLQ,消费消息。
二、 延迟队列实现
队列中的消息成为死信(Dead Letter)的几种情况:
1) The message is rejected (basic.reject or basic.nack) with requeue=false,
2) The TTL for the message expires; or
3) The queue length limit is exceeded.
而通过配置队列的x-dead-letter-exchange及x-dead-letter-routing-key键值,Dead Letter就会被重新发送到指定的DLX中。
如上图所示,注意延迟队列并没有消费者,并且设置参数:
x-dead-letter-exchange:DLX
x-dead-letter-routing-key:DLK
x-max-length:1000000 // length = 1million
x-message-ttl:30000 // TTL = 30s
生产者发布的消息路由到该延迟队列,30秒后消息成为死信并被重新发送到死信交换器DLX中;消费者订阅死信队列DLQ,消费消息。
二、 延迟队列实现
-
-
"classpath:rabbitmq-cfg.properties")(
-
public class RabbitConfig {
-
/**
-
* 配置RabbitMQ连接工厂
-
*
-
* @param host
-
* @param port
-
* @param username
-
* @param password
-
* @param virtualHost
-
* @param connectionTimeout
-
* @return
-
*/
-
-
public ConnectionFactory connectionFactory(@Value("${spring.rabbitmq.host}") String host,
-
@Value("${spring.rabbitmq.port}") String port, @Value("${spring.rabbitmq.username}") String username,
-
@Value("${spring.rabbitmq.password}") String password,
-
@Value("${spring.rabbitmq.virtual-host}") String virtualHost,
-
@Value("${spring.rabbitmq.connection-timeout}") String connectionTimeout) {
-
CachingConnectionFactory ret = new CachingConnectionFactory();
-
ret.setHost(host);
-
ret.setPort(Integer.parseInt(port));
-
ret.setUsername(username);
-
ret.setPassword(password);
-
ret.setVirtualHost(virtualHost);
-
ret.setConnectionTimeout(Integer.parseInt(connectionTimeout));
-
return ret;
-
}
-
-
/**
-
* 声明30s延迟队列
-
*
-
* @param name
-
* @return
-
*/
-
-
public Queue queueDelay30s(@Value("${atf.queue.delay.30s}") String name) {
-
Map<String, Object> arguments = new HashMap<String, Object>();
-
arguments.put( "x-message-ttl", 30 * 1000); // Message TTL = 30s
-
arguments.put( "x-max-length", 1000000); // Max length = 1million
-
// 死信路由到死信交换器DLX
-
arguments.put( "x-dead-letter-exchange", "atf.exchange.delay.dlx");
-
arguments.put( "x-dead-letter-routing-key", "atf.routingkey.delay.dlk");
-
return new Queue(name, true, false, false, arguments);
-
}
-
-
/**
-
* 声明死信队列DLQ
-
*
-
* @param name
-
* @return
-
*/
-
-
public Queue queueDelayDlq(@Value("${atf.queue.delay.dlq}") String name) {
-
return new Queue(name, true, false, false);
-
}
-
-
/**
-
* 声明延迟交换器
-
*
-
* @param name
-
* @return
-
*/
-
-
public DirectExchange exchangeDelay(@Value("${atf.exchange.delay}") String name) {
-
DirectExchange ret = new DirectExchange(name, true, false);
-
ret.setInternal( false);
-
return ret;
-
}
-
-
/**
-
* 声明死信交换器DLX
-
*
-
* @param name
-
* @return
-
*/
-
-
public DirectExchange exchangeDelayDlx(@Value("${atf.exchange.delay.dlx}") String name) {
-
return new DirectExchange(name, true, false);
-
}
-
-
/**
-
* 绑定延迟交换器和30s延迟队列
-
*
-
* @param queue
-
* @param exchange
-
* @param routingKey
-
* @return
-
*/
-
-
public Binding bindingDelay(@Qualifier("queueDelay30s") Queue queue,
-
@Qualifier("exchangeDelay") DirectExchange exchange,
-
@Value("${atf.routingkey.delay.30s}") String routingKey) {
-
return BindingBuilder.bind(queue).to(exchange).with(routingKey);
-
}
-
-
/**
-
* 绑定死信交换器DLX和死信队列DLQ
-
*
-
* @param queue
-
* @param exchange
-
* @param routingKey
-
* @return
-
*/
-
-
public Binding bindingDelayConsume(@Qualifier("queueDelayDlq") Queue queue,
-
@Qualifier("exchangeDelayDlx") DirectExchange exchange,
-
@Value("${atf.routingkey.delay.dlk}") String routingKey) {
-
return BindingBuilder.bind(queue).to(exchange).with(routingKey);
-
}
-
}
-
public void sendDelay(AtfEventPayload payload) {
-
rabbitTemplate.convertAndSend( "atf.exchange.delay", "atf.routingkey.delay.30s", payload);
-
}
-
"atf.queue.delay.dlq", containerFactory = "rabbitListenerContainerFactory")(queues =
-
public void consumeDelay(AtfEventPayload payload) {
-
try {
-
AtWorkflow workflow = workflowSvs.find(payload.getWorkflowName());
-
workflowSvs.execute(workflow, null);
-
} catch (Exception e) {
-
e.printStackTrace();
-
}
-
}
像支付宝,微信支付后的回调机制,可以使用延迟队列实现。可是创建延时不同时间的队列,死信后被消费,消费异常后,消息进入不同的队列。
/** * 默认及时消息交换机 * @return */ @Bean() public DirectExchange defaultDirectExchange() { return new DirectExchange(MessageQueueConstants.DEFAULT_DIRECT_EXCHANGE_NAME, true, false); } /** * 默认延迟消息死信队列 * @return */ @Bean public Queue defaultDeadLetterQueue10() { Map<String, Object> arguments = new HashMap<>(); arguments.put("x-dead-letter-exchange",MessageQueueConstants.DEFAULT_DIRECT_EXCHANGE_NAME);//设置交换机路由 arguments.put("x-dead-letter-routing-key", MessageQueueConstants.DEFAULT_REPEAT_TRADE_QUEUE_NAME);//设置转发队列名称 arguments.put("x-message-ttl", 10 * 1000); // Message TTL = 10s Queue queue = new Queue(MessageQueueConstants.DEFAULT_DEAD_LETTER_QUEUE_NAME10,true,false,false,arguments); return queue; } /** * 默认延迟消息死信队列 * @return */ @Bean public Queue defaultDeadLetterQueue30() { Map<String, Object> arguments = new HashMap<>(); arguments.put("x-dead-letter-exchange",MessageQueueConstants.DEFAULT_DIRECT_EXCHANGE_NAME);//设置交换机路由 arguments.put("x-dead-letter-routing-key", MessageQueueConstants.DEFAULT_REPEAT_TRADE_QUEUE_NAME);//设置转发队列名称 arguments.put("x-message-ttl", 30 * 1000); // Message TTL = 30s Queue queue = new Queue(MessageQueueConstants.DEFAULT_DEAD_LETTER_QUEUE_NAME30,true,false,false,arguments); return queue; } /** * 默认延迟消息死信队列 * @return */ @Bean public Queue defaultDeadLetterQueue180() { Map<String, Object> arguments = new HashMap<>(); arguments.put("x-dead-letter-exchange",MessageQueueConstants.DEFAULT_DIRECT_EXCHANGE_NAME);//设置交换机路由 arguments.put("x-dead-letter-routing-key", MessageQueueConstants.DEFAULT_REPEAT_TRADE_QUEUE_NAME);//设置转发队列名称 arguments.put("x-message-ttl", 1000*60*3); // Message TTL = 180s Queue queue = new Queue(MessageQueueConstants.DEFAULT_DEAD_LETTER_QUEUE_NAME180,true,false,false,arguments); return queue; } /** * 默认延迟消息死信队列 * @return */ @Bean public Queue defaultDeadLetterQueue1200() { Map<String, Object> arguments = new HashMap<>(); arguments.put("x-dead-letter-exchange",MessageQueueConstants.DEFAULT_DIRECT_EXCHANGE_NAME);//设置交换机路由 arguments.put("x-dead-letter-routing-key", MessageQueueConstants.DEFAULT_REPEAT_TRADE_QUEUE_NAME);//设置转发队列名称 arguments.put("x-message-ttl", 1000*60*20); // Message TTL = 1200s Queue queue = new Queue(MessageQueueConstants.DEFAULT_DEAD_LETTER_QUEUE_NAME1200,true,false,false,arguments); return queue; } /** * 默认延迟消息死信队列 * @return */ @Bean public Queue defaultDeadLetterQueue3600() { Map<String, Object> arguments = new HashMap<>(); arguments.put("x-dead-letter-exchange",MessageQueueConstants.DEFAULT_DIRECT_EXCHANGE_NAME);//设置交换机路由 arguments.put("x-dead-letter-routing-key", MessageQueueConstants.DEFAULT_REPEAT_TRADE_QUEUE_NAME);//设置转发队列名称 arguments.put("x-message-ttl", 1000*60*60); // Message TTL = 3600s Queue queue = new Queue(MessageQueueConstants.DEFAULT_DEAD_LETTER_QUEUE_NAME3600,true,false,false,arguments); return queue; } @Bean public Binding defaultDeadLetterBinding10() { Binding bind = BindingBuilder.bind(defaultDeadLetterQueue10()).to(defaultDirectExchange()).with(MessageQueueConstants.DEFAULT_DEAD_LETTER_QUEUE_NAME10); return bind; } @Bean public Binding defaultDeadLetterBinding30() { Binding bind = BindingBuilder.bind(defaultDeadLetterQueue30()).to(defaultDirectExchange()).with(MessageQueueConstants.DEFAULT_DEAD_LETTER_QUEUE_NAME30); return bind; } @Bean public Binding defaultDeadLetterBinding180() { Binding bind = BindingBuilder.bind(defaultDeadLetterQueue180()).to(defaultDirectExchange()).with(MessageQueueConstants.DEFAULT_DEAD_LETTER_QUEUE_NAME180); return bind; } @Bean public Binding defaultDeadLetterBinding1200() { Binding bind = BindingBuilder.bind(defaultDeadLetterQueue1200()).to(defaultDirectExchange()).with(MessageQueueConstants.DEFAULT_DEAD_LETTER_QUEUE_NAME1200); return bind; } @Bean public Binding defaultDeadLetterBinding3600() { Binding bind = BindingBuilder.bind(defaultDeadLetterQueue3600()).to(defaultDirectExchange()).with(MessageQueueConstants.DEFAULT_DEAD_LETTER_QUEUE_NAME3600); return bind; } /** * 默认延迟消息死信接受转发消息队列 * @return */ @Bean public Queue defaultRepeatTradeQueue() { Queue queue = new Queue(MessageQueueConstants.DEFAULT_REPEAT_TRADE_QUEUE_NAME,true,false,false); return queue; } @Bean public Binding defaultRepeatTradeBinding() { return BindingBuilder.bind(defaultRepeatTradeQueue()).to(defaultDirectExchange()).with(MessageQueueConstants.DEFAULT_REPEAT_TRADE_QUEUE_NAME); }