延時隊列顧名思義,即放置在該隊列裏面的消息是不須要當即消費的,而是等待一段時間以後取出消費。
那麼,爲何須要延遲消費呢?咱們來看如下的場景html
訂單業務: 在電商/點餐中,都有下單後 30 分鐘內沒有付款,就自動取消訂單。
短信通知: 下單成功後 60s 以後給用戶發送短信通知。
失敗重試: 業務操做失敗後,間隔必定的時間進行失敗重試。java
本文基於springboot,使用rabbitmq_delayed_message_exchange插件實現延時隊列(RabbitMQ及其插件環境安裝點此),具體實踐以下:web
application.propertiesspring
spring.rabbitmq.username=root spring.rabbitmq.password=root spring.rabbitmq.host=192.168.1.123 spring.rabbitmq.port=5672 spring.rabbitmq.virtual-host=test
XdelayConfig.javaspringboot
import org.springframework.amqp.core.Binding; import org.springframework.amqp.core.BindingBuilder; import org.springframework.amqp.core.CustomExchange; import org.springframework.amqp.core.Queue; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import java.util.HashMap; import java.util.Map; @Configuration public class XdelayConfig { /** * 當即消費的隊列名稱 */ public static final String IMMEDIATE_QUEUE_XDELAY = "queue.xdelay.immediate"; /** * 延時的exchange */ public static final String DELAYED_EXCHANGE_XDELAY = "exchange.xdelay.delayed"; public static final String DELAY_ROUTING_KEY_XDELAY = "routingkey.xdelay.delay"; /** * 建立一個當即消費隊列 * * @return */ @Bean public Queue immediateQueue() { // 第一個參數是建立的queue的名字,第二個參數是是否支持持久化 return new Queue(IMMEDIATE_QUEUE_XDELAY, true); } @Bean public CustomExchange delayExchange() { Map<String, Object> args = new HashMap<String, Object>(); args.put("x-delayed-type", "direct"); return new CustomExchange(DELAYED_EXCHANGE_XDELAY, "x-delayed-message", true, false, args); } /** * 把當即消費的隊列和延時消費的exchange綁定在一塊兒 * * @return */ @Bean public Binding bindingNotify() { return BindingBuilder.bind(immediateQueue()).to(delayExchange()).with(DELAY_ROUTING_KEY_XDELAY).noargs(); } }
XdelaySender.java 生產者app
import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.amqp.rabbit.core.RabbitTemplate; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; import java.text.SimpleDateFormat; import java.util.Date; /** * 生產者 */ @Component public class XdelaySender { private final static Logger logger = LoggerFactory.getLogger(XdelaySender.class); @Autowired private RabbitTemplate rabbitTemplate; public void send(String msg, int delayTime) { logger.info("msg= " + msg + ".delayTime" + delayTime); SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); this.rabbitTemplate.convertAndSend(XdelayConfig.DELAYED_EXCHANGE_XDELAY, XdelayConfig.DELAY_ROUTING_KEY_XDELAY, msg, message -> { message.getMessageProperties().setDelay(delayTime); System.out.println(sdf.format(new Date()) + " Delay sent."); return message; }); } }
XdelayReceiver.java 消費者微服務
import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.amqp.rabbit.annotation.EnableRabbit; import org.springframework.amqp.rabbit.annotation.RabbitListener; import org.springframework.context.annotation.Configuration; import org.springframework.stereotype.Component; import java.text.SimpleDateFormat; import java.util.Date; /** * 消費者 */ @Component @EnableRabbit @Configuration public class XdelayReceiver { private final static Logger logger = LoggerFactory.getLogger(XdelayReceiver.class); @RabbitListener(queues = com.example.antchat.rabbitmq.XdelayConfig.IMMEDIATE_QUEUE_XDELAY) public void get(String msg) { SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); logger.info("收到延時消息時間:" + sdf.format(new Date()) + " Delay sent."); logger.info("收到延時消息:" + msg); } }
測試測試
import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RestController; @RestController public class RabbitMQController { @Autowired XdelaySender xdelaySender; @RequestMapping("/testRabbit") public void testRabbit() { xdelaySender.send("我來發一個測試消息,10秒", 10000);//10秒 xdelaySender.send("我來發一個測試消息,2秒", 2000);//2秒 xdelaySender.send("我來發一個測試消息,1秒", 2000);//1秒 } }
參考博文:ui