- 資訊首頁(yè) > 開(kāi)發(fā)技術(shù) >
- Springboot+rabbitmq實(shí)現延時(shí)隊列的兩種方式
延時(shí)隊列顧名思義,即放置在該隊列里面的消息是不需要立即消費的,而是等待一段時(shí)間之后取出消費。
那么,為什么需要延遲消費呢?我們來(lái)看以下的場(chǎng)景
這些場(chǎng)景都非常常見(jiàn),我們可以思考,比如第二個(gè)需求,系統創(chuàng )建了預約之后,需要在預約時(shí)間到達前一小時(shí)提醒被預約的雙方參會(huì )。那么一天之中肯定是會(huì )有很多個(gè)預約的,時(shí)間也是不一定的,假設現在有1點(diǎn) 2點(diǎn) 3點(diǎn) 三個(gè)預約,如何讓系統知道在當前時(shí)間等于0點(diǎn) 1點(diǎn) 2點(diǎn)給用戶(hù)發(fā)送信息呢,是不是需要一個(gè)輪詢(xún),一直去查看所有的預約,比對當前的系統時(shí)間和預約提前一小時(shí)的時(shí)間是否相等呢?這樣做非常浪費資源而且輪詢(xún)的時(shí)間間隔不好控制。如果我們使用延時(shí)消息隊列呢,我們在創(chuàng )建時(shí)把需要通知的預約放入消息中間件中,并且設置該消息的過(guò)期時(shí)間,等過(guò)期時(shí)間到達時(shí)再取出消費即可。
Rabbitmq實(shí)現延時(shí)隊列一般而言有兩種形式:
第一種方式:利用兩個(gè)特性: Time To Live(TTL)、Dead Letter Exchanges(DLX)
第二種方式:利用rabbitmq中的插件x-delay-message
TTL
RabbitMQ可以針對隊列設置x-expires(則隊列中所有的消息都有相同的過(guò)期時(shí)間)或者針對Message設置x-message-ttl(對消息進(jìn)行單獨設置,每條消息TTL可以不同),來(lái)控制消息的生存時(shí)間,如果超時(shí)(兩者同時(shí)設置以最先到期的時(shí)間為準),則消息變?yōu)閐ead letter(死信)
Dead Letter Exchanges(DLX)
RabbitMQ的Queue可以配置x-dead-letter-exchange和x-dead-letter-routing-key(可選)兩個(gè)參數,如果隊列內出現了dead letter,則按照這兩個(gè)參數重新路由轉發(fā)到指定的隊列。
x-dead-letter-exchange:出現dead letter之后將dead letter重新發(fā)送到指定exchange
x-dead-letter-routing-key:出現dead letter之后將dead letter重新按照指定的routing-key發(fā)送
在pom.xml文件中增加rabbitmq的依賴(lài)
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-amqp</artifactId> </dependency>
初始化queue exchange和queue及exchange之間的binding關(guān)系
Config.java
package com.example.demo.deadLetter; import java.util.HashMap; import java.util.Map; import org.springframework.amqp.core.Binding; import org.springframework.amqp.core.BindingBuilder; import org.springframework.amqp.core.DirectExchange; import org.springframework.amqp.core.Queue; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import com.example.demo.Constants.Constants; @Configuration public class Config { // 創(chuàng )建一個(gè)立即消費隊列 @Bean public Queue immediateQueue() { // 第一個(gè)參數是創(chuàng )建的queue的名字,第二個(gè)參數是是否支持持久化 return new Queue(Constants.IMMEDIATE_QUEUE, true); } // 創(chuàng )建一個(gè)延時(shí)隊列 @Bean public Queue delayQueue() { Map<String, Object> params = new HashMap<>(); // x-dead-letter-exchange 聲明了隊列里的死信轉發(fā)到的DLX名稱(chēng), params.put("x-dead-letter-exchange", Constants.IMMEDIATE_EXCHANGE); // x-dead-letter-routing-key 聲明了這些死信在轉發(fā)時(shí)攜帶的 routing-key 名稱(chēng)。 params.put("x-dead-letter-routing-key", Constants.IMMEDIATE_ROUTING_KEY); return new Queue(Constants.DELAY_QUEUE, true, false, false, params); } @Bean public DirectExchange immediateExchange() { // 一共有三種構造方法,可以只傳exchange的名字, 第二種,可以傳exchange名字,是否支持持久化,是否可以自動(dòng)刪除, //第三種在第二種參數上可以增加Map,Map中可以存放自定義exchange中的參數 return new DirectExchange(Constants.IMMEDIATE_EXCHANGE, true, false); } @Bean public DirectExchange deadLetterExchange() { // 一共有三種構造方法,可以只傳exchange的名字, 第二種,可以傳exchange名字,是否支持持久化,是否可以自動(dòng)刪除, //第三種在第二種參數上可以增加Map,Map中可以存放自定義exchange中的參數 return new DirectExchange(Constants.DEAD_LETTER_EXCHANGE, true, false); } @Bean //把立即消費的隊列和立即消費的exchange綁定在一起 public Binding immediateBinding() { return BindingBuilder.bind(immediateQueue()).to(immediateExchange()).with(Constants.IMMEDIATE_ROUTING_KEY); } @Bean //把立即消費的隊列和立即消費的exchange綁定在一起 public Binding delayBinding() { return BindingBuilder.bind(delayQueue()).to(deadLetterExchange()).with(Constants.DELAY_ROUTING_KEY); } }
生產(chǎn)者生產(chǎn)消息
ImmediateSender.java
package com.example.demo.deadLetter; import java.text.SimpleDateFormat; import java.util.Date; import org.springframework.amqp.rabbit.core.RabbitTemplate; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; import com.example.demo.Constants.Constants; import com.example.demo.model.Booking; @Component public class ImmediateSender { @Autowired private RabbitTemplate rabbitTemplate; public void send(Booking booking, int delayTime) { System.out.println("delayTime" + delayTime); SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); this.rabbitTemplate.convertAndSend(Constants.DEAD_LETTER_EXCHANGE, Constants.DELAY_ROUTING_KEY, booking, message -> { message.getMessageProperties().setExpiration(delayTime + ""); System.out.println(sdf.format(new Date()) + " Delay sent."); return message; }); } }
消費者消費消息
ImmediateReceiver.java
package com.example.demo.deadLetter; import org.springframework.amqp.rabbit.annotation.EnableRabbit; import org.springframework.amqp.rabbit.annotation.RabbitHandler; import org.springframework.amqp.rabbit.annotation.RabbitListener; import org.springframework.context.annotation.Configuration; import org.springframework.stereotype.Component; import com.example.demo.Constants.Constants; import com.example.demo.model.Booking; @Component @EnableRabbit @Configuration public class ImmediateReceiver { @RabbitListener(queues = Constants.IMMEDIATE_QUEUE) @RabbitHandler public void get(Booking booking) { System.out.println("收到延時(shí)消息了" + booking); } }
model類(lèi)book
Book.java
package com.example.demo.model; import java.io.Serializable; import java.util.Date; public class Booking implements Serializable { private static final long serialVersionUID = 1L; private String bookingName; private Date bookingTime; private String bookingContent; private String operatorName; public Booking() { } public String getBookingName() { return bookingName; } public void setBookingName(String bookingName) { this.bookingName = bookingName; } public Date getBookingTime() { return bookingTime; } public void setBookingTime(Date bookingTime) { this.bookingTime = bookingTime; } public String getBookingContent() { return bookingContent; } public void setBookingContent(String bookingContent) { this.bookingContent = bookingContent; } public String getOperatorName() { return operatorName; } public void setOperatorName(String operatorName) { this.operatorName = operatorName; } @Override public String toString() { return super.toString(); } }
測試類(lèi)
Test.java
package com.example.demo; import java.util.Date; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.test.context.junit4.SpringRunner; import com.example.demo.Immediate.Sender; import com.example.demo.deadLetter.ImmediateSender; import com.example.demo.model.Booking; @RunWith(SpringRunner.class) @SpringBootTest public class RabbitMqTestApplicationTests { @Autowired ImmediateSender immediateSender; @Test public void test() { Booking booking = new Booking(); booking.setBookingContent("hhaha"); booking.setBookingName("預定房子"); booking.setBookingTime(new Date()); booking.setOperatorName("hellen"); immediateSender.send(booking, 1000); } }
總結第一種方式:經(jīng)過(guò)測試,我們可以發(fā)現,當我們先增加一條過(guò)期時(shí)間大(10000)的A消息進(jìn)入,之后再增加一個(gè)過(guò)期時(shí)間小的(1000)消息B,并沒(méi)有出現想象中的B消息先被消費,A消息后被消費,而是出現了當10000過(guò)去的時(shí)候,AB消息同時(shí)被消費,也就是B消息的消費被阻塞了。
為什么會(huì )出現這樣的現象呢?
我們知道利用TTL DLX特性實(shí)現的方式,實(shí)際上在第一個(gè)延時(shí)隊列C里面設置了dlx,生產(chǎn)者生產(chǎn)了一條帶ttl的消息放入了延時(shí)隊列C中,等到延時(shí)時(shí)間到了,延時(shí)隊列C中的消息變成了死信,根據延時(shí)隊列C中設置的dlx的exchange的轉發(fā)規則,轉發(fā)到了實(shí)際消費隊列D中,當該隊列中的監聽(tīng)器監聽(tīng)到消息時(shí)就會(huì )正式開(kāi)始消費。那么實(shí)際上延時(shí)隊列中的消息也是放入隊列中的,隊列滿(mǎn)足先進(jìn)先出,而延時(shí)大的消息A還沒(méi)出隊,所以B消息也不能順利出隊。
為了解決上面的問(wèn)題,Rabbitmq實(shí)現了一個(gè)插件x-delay-message來(lái)實(shí)現延時(shí)隊列。
介紹Ubuntu系統下插件安裝方式:
選擇插件,選擇3.6版本,進(jìn)行下載
將安裝包進(jìn)行解壓
uzip rabbitmq_delayed_message_exchange-20171215-3.6.x.zip
將插件移到rabbitmq安裝的路徑
sudo cp -r rabbitmq_delayed_message_exchange-20171215-3.6.x.ez /usr/lib/rabbitmq/lib/rabbitmq_server-3.6.15/plugins
Enable插件
rabbitmq-plugins enable rabbitmq_delayed_message_exchange
windows同理
XdelayConfig.java
package com.example.demo.Xdelay; import java.util.HashMap; import java.util.Map; 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 com.example.demo.Constants.Constants; @Configuration public class XdelayConfig { // 創(chuàng )建一個(gè)立即消費隊列 @Bean public Queue immediateQueue() { // 第一個(gè)參數是創(chuàng )建的queue的名字,第二個(gè)參數是是否支持持久化 return new Queue(Constants.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(Constants.DELAYED_EXCHANGE_XDELAY, "x-delayed-message", true, false, args); } @Bean public Binding bindingNotify() { return BindingBuilder.bind(immediateQueue()).to(delayExchange()).with(Constants.DELAY_ROUTING_KEY_XDELAY).noargs(); } }
XdelaySender.java
package com.example.demo.Xdelay; import java.text.SimpleDateFormat; import java.util.Date; import org.springframework.amqp.AmqpException; import org.springframework.amqp.core.Message; import org.springframework.amqp.core.MessagePostProcessor; import org.springframework.amqp.rabbit.core.RabbitTemplate; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import com.example.demo.Constants.Constants; import com.example.demo.model.Booking; @Service public class XdelaySender { @Autowired private RabbitTemplate rabbitTemplate; public void send(Booking booking, int delayTime) { System.out.println("delayTime" + delayTime); SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); this.rabbitTemplate.convertAndSend(Constants.DELAYED_EXCHANGE_XDELAY, Constants.DELAY_ROUTING_KEY_XDELAY, booking, new MessagePostProcessor() { @Override public Message postProcessMessage(Message message) throws AmqpException { message.getMessageProperties().setDelay(delayTime); System.out.println(sdf.format(new Date()) + " Delay sent."); return message; } }); } }
XdelayReceiver.java
package com.example.demo.Xdelay; 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 com.example.demo.Constants.Constants; import com.example.demo.model.Booking; @Component @EnableRabbit @Configuration public class XdelayReceiver { @RabbitListener(queues = Constants.IMMEDIATE_QUEUE_XDELAY) public void get(Booking booking) { System.out.println("Receive" + booking); } }
Test.java
package com.example.demo; import java.util.Date; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.test.context.junit4.SpringRunner; import com.example.demo.Xdelay.XdelaySender; import com.example.demo.model.Booking; @RunWith(SpringRunner.class) @SpringBootTest public class RabbitMqTestApplicationTests { @Autowired XdelaySender xdelaySender; @Test public void test11() { Booking booking = new Booking(); booking.setBookingContent("hhaha"); booking.setBookingName("預定房子"); booking.setBookingTime(new Date()); booking.setOperatorName("hellen"); xdelaySender.send(booking, 2000); } }
到此這篇關(guān)于Springboot+rabbitmq實(shí)現延時(shí)隊列的兩種方式的文章就介紹到這了,更多相關(guān)Springboot rabbitmq延時(shí)隊列內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng )、來(lái)自互聯(lián)網(wǎng)轉載和分享為主,文章觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權請聯(lián)系QQ:712375056 進(jìn)行舉報,并提供相關(guān)證據,一經(jīng)查實(shí),將立刻刪除涉嫌侵權內容。
Copyright ? 2009-2021 56dr.com. All Rights Reserved. 特網(wǎng)科技 特網(wǎng)云 版權所有 珠海市特網(wǎng)科技有限公司 粵ICP備16109289號
域名注冊服務(wù)機構:阿里云計算有限公司(萬(wàn)網(wǎng)) 域名服務(wù)機構:煙臺帝思普網(wǎng)絡(luò )科技有限公司(DNSPod) CDN服務(wù):阿里云計算有限公司 中國互聯(lián)網(wǎng)舉報中心 增值電信業(yè)務(wù)經(jīng)營(yíng)許可證B2
建議您使用Chrome、Firefox、Edge、IE10及以上版本和360等主流瀏覽器瀏覽本網(wǎng)站