日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Springboot+rabbitmq實現延時隊列的兩種方式

瀏覽:251日期:2023-03-12 08:31:18
目錄什么是延時隊列,延時隊列應用于什么場景

延時隊列顧名思義,即放置在該隊列里面的消息是不需要立即消費的,而是等待一段時間之后取出消費。那么,為什么需要延遲消費呢?我們來看以下的場景

網上商城下訂單后30分鐘后沒有完成支付,取消訂單(如:淘寶、去哪兒網) 系統創建了預約之后,需要在預約時間到達前一小時提醒被預約的雙方參會 系統中的業務失敗之后,需要重試

這些場景都非常常見,我們可以思考,比如第二個需求,系統創建了預約之后,需要在預約時間到達前一小時提醒被預約的雙方參會。那么一天之中肯定是會有很多個預約的,時間也是不一定的,假設現在有1點 2點 3點 三個預約,如何讓系統知道在當前時間等于0點 1點 2點給用戶發送信息呢,是不是需要一個輪詢,一直去查看所有的預約,比對當前的系統時間和預約提前一小時的時間是否相等呢?這樣做非常浪費資源而且輪詢的時間間隔不好控制。如果我們使用延時消息隊列呢,我們在創建時把需要通知的預約放入消息中間件中,并且設置該消息的過期時間,等過期時間到達時再取出消費即可。

Rabbitmq實現延時隊列一般而言有兩種形式:第一種方式:利用兩個特性: Time To Live(TTL)、Dead Letter Exchanges(DLX)第二種方式:利用rabbitmq中的插件x-delay-message

利用TTL DLX實現延時隊列的方式TTL DLX是什么

TTL

RabbitMQ可以針對隊列設置x-expires(則隊列中所有的消息都有相同的過期時間)或者針對Message設置x-message-ttl(對消息進行單獨設置,每條消息TTL可以不同),來控制消息的生存時間,如果超時(兩者同時設置以最先到期的時間為準),則消息變為dead letter(死信)

Dead Letter Exchanges(DLX)RabbitMQ的Queue可以配置x-dead-letter-exchange和x-dead-letter-routing-key(可選)兩個參數,如果隊列內出現了dead letter,則按照這兩個參數重新路由轉發到指定的隊列。x-dead-letter-exchange:出現dead letter之后將dead letter重新發送到指定exchangex-dead-letter-routing-key:出現dead letter之后將dead letter重新按照指定的routing-key發送

Springboot集成rabbitmq實現第一種方式

在pom.xml文件中增加rabbitmq的依賴

<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-amqp</artifactId></dependency>

初始化queue exchange和queue及exchange之間的binding關系

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;@Configurationpublic class Config { // 創建一個立即消費隊列 @Bean public Queue immediateQueue() {// 第一個參數是創建的queue的名字,第二個參數是是否支持持久化return new Queue(Constants.IMMEDIATE_QUEUE, true); } // 創建一個延時隊列 @Bean public Queue delayQueue() {Map<String, Object> params = new HashMap<>();// x-dead-letter-exchange 聲明了隊列里的死信轉發到的DLX名稱,params.put('x-dead-letter-exchange', Constants.IMMEDIATE_EXCHANGE);// x-dead-letter-routing-key 聲明了這些死信在轉發時攜帶的 routing-key 名稱。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名字,是否支持持久化,是否可以自動刪除,//第三種在第二種參數上可以增加Map,Map中可以存放自定義exchange中的參數return new DirectExchange(Constants.IMMEDIATE_EXCHANGE, true, false); } @Bean public DirectExchange deadLetterExchange() {// 一共有三種構造方法,可以只傳exchange的名字, 第二種,可以傳exchange名字,是否支持持久化,是否可以自動刪除,//第三種在第二種參數上可以增加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); }}

生產者生產消息

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;@Componentpublic 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@Configurationpublic class ImmediateReceiver { @RabbitListener(queues = Constants.IMMEDIATE_QUEUE) @RabbitHandler public void get(Booking booking) {System.out.println('收到延時消息了' + booking); }}

model類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(); }}

測試類

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)@SpringBootTestpublic 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); }}

總結第一種方式:經過測試,我們可以發現,當我們先增加一條過期時間大(10000)的A消息進入,之后再增加一個過期時間小的(1000)消息B,并沒有出現想象中的B消息先被消費,A消息后被消費,而是出現了當10000過去的時候,AB消息同時被消費,也就是B消息的消費被阻塞了。

為什么會出現這樣的現象呢?

我們知道利用TTL DLX特性實現的方式,實際上在第一個延時隊列C里面設置了dlx,生產者生產了一條帶ttl的消息放入了延時隊列C中,等到延時時間到了,延時隊列C中的消息變成了死信,根據延時隊列C中設置的dlx的exchange的轉發規則,轉發到了實際消費隊列D中,當該隊列中的監聽器監聽到消息時就會正式開始消費。那么實際上延時隊列中的消息也是放入隊列中的,隊列滿足先進先出,而延時大的消息A還沒出隊,所以B消息也不能順利出隊。

利用Rabbitmq的插件x-delay-message實現

為了解決上面的問題,Rabbitmq實現了一個插件x-delay-message來實現延時隊列。

x-delay-message安裝

介紹Ubuntu系統下插件安裝方式:選擇rabbitmq_delayed_message_exchange插件,選擇3.6版本,進行下載將安裝包進行解壓

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同理

Springboot集成rabbitmq實現第二種方式

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;@Configurationpublic class XdelayConfig { // 創建一個立即消費隊列 @Bean public Queue immediateQueue() {// 第一個參數是創建的queue的名字,第二個參數是是否支持持久化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;@Servicepublic 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@Configurationpublic 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)@SpringBootTestpublic 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); }}

到此這篇關于Springboot+rabbitmq實現延時隊列的兩種方式的文章就介紹到這了,更多相關Springboot rabbitmq延時隊列內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Spring
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
欧美日韩亚洲国产精品| 蜜臀久久精品| 麻豆久久精品| 久久精品亚洲人成影院| 桃色一区二区| 亚洲韩日在线| 欧美韩一区二区| 激情亚洲影院在线观看| 久久精品一区二区三区中文字幕| 欧美在线不卡| 成人在线视频中文字幕| 欧美日韩四区| 国产一区二区久久久久| 欧美日韩免费观看一区=区三区| 久久亚洲欧美| 亚洲色诱最新| 日韩欧美三区| 里番精品3d一二三区| 国产精品毛片久久| 伊人久久大香线蕉av不卡| 国产手机视频一区二区| 亚洲精品在线国产| 国产情侣久久| 精品视频网站| 国产综合婷婷| 亚洲一级大片| 麻豆91在线播放| 国内精品福利| 色综合视频一区二区三区日韩| 国产精品美女午夜爽爽| 日韩理论片av| 伊人国产精品| 欧美国产日本| 韩日一区二区三区| 日韩高清国产一区在线| 国产一区二区三区不卡av| 欧美福利在线| 欧美日韩18| 天堂а√在线最新版中文在线| 久久激情婷婷| 日本午夜免费一区二区 | 日韩1区2区日韩1区2区| 一级欧美视频| 欧美午夜不卡| 国产成人精品一区二区三区视频| 国产精品最新| 国产精品尤物| 国产欧美日韩精品一区二区免费 | 国产探花在线精品一区二区| 久久久久一区| 亚洲天堂av影院| 午夜精品免费| 国产探花在线精品一区二区| 日韩中文视频| 日韩高清一级| 激情综合自拍| 美女久久久精品| 国产亚洲在线观看| 精品国产精品国产偷麻豆 | 精品国产一区二区三区2021| 国产精品99免费看| 亚洲专区一区| 婷婷色综合网| 亚洲视频国产| 国产精品a级| 久久精品亚洲人成影院| 一区二区91| 久久爱www.| jizzjizz中国精品麻豆| 亚洲一区二区三区四区电影| 91精品美女| 欧洲毛片在线视频免费观看| 高潮久久久久久久久久久久久久| 国产丝袜一区| 首页亚洲欧美制服丝腿| 美女精品在线观看| 国产成人免费| 日本国产一区| 一区在线免费观看| 国产精品成久久久久| 日韩av一区二区在线影视| 久久精品国产大片免费观看| 欧美日韩午夜| 日韩精品一区第一页| 日韩免费高清| 久久国产88| 国产高清不卡| 国产精品一站二站| 播放一区二区| 中文字幕一区日韩精品| 国产不卡av一区二区| 91国语精品自产拍| 国产亚洲一区| 久久国产精品成人免费观看的软件| 国精品一区二区| 精品日本视频| 天堂va在线高清一区| 在线亚洲一区| 欧产日产国产精品视频| 国产精品成人国产| 亚洲精品在线二区| 久久网站免费观看| 在线人成日本视频| 欧美亚洲一级| 国产精品久久久久久久免费观看| 欧美日韩免费看片| 国产传媒av在线| 精品成人免费一区二区在线播放| 一区福利视频| 精品国产精品国产偷麻豆| 美女精品久久| 久久av电影| 欧美日韩一区二区三区在线电影| 在线观看亚洲精品福利片| 欧美日韩国产精品一区二区亚洲| 日韩国产一区二区三区| 色在线中文字幕| 亚洲涩涩在线| 黄色精品视频| 蜜臀久久久99精品久久久久久| 尤物网精品视频| 蜜桃成人av| 999国产精品| 91精品国产福利在线观看麻豆| 国产精品白丝一区二区三区| 国产欧美69| 精品中文在线| 国产欧美一区二区三区精品观看 | 欧美sm一区| 免费在线观看一区| 麻豆一区在线| 亚洲一级影院| 欧美日韩国产一区精品一区| 日韩中文字幕av电影| 鲁大师成人一区二区三区| 久久国产生活片100| 日韩激情啪啪| 婷婷精品在线观看| 国产成人精品一区二区三区视频| 综合激情一区| 国产亚洲精品v| 蜜桃国内精品久久久久软件9| 激情欧美丁香| 亚洲欧美日韩视频二区| 亚洲我射av| 极品裸体白嫩激情啪啪国产精品| 国产综合精品一区| 免费人成在线不卡| 亚洲大片在线| 日韩欧乱色一区二区三区在线| 成人国产精品久久| 清纯唯美亚洲综合一区| 成人欧美一区二区三区的电影| 欧美特黄a级高清免费大片a级| 国产精品宾馆| 国产精品美女久久久| 欧美91在线| 亚洲影院天堂中文av色| 国产一区调教| 在线观看亚洲精品福利片| 国产精品一区二区美女视频免费看 | 国产精品日韩精品在线播放| 日韩av电影一区| 久久精品国产大片免费观看| 久久精品电影| 欧美综合另类| 蜜桃视频一区二区三区在线观看| 亚洲国产日韩欧美在线| 日韩精品久久久久久久电影99爱| 亚洲精品美女91| 久久国内精品| 国产成人精品三级高清久久91| 欧美日韩在线观看首页| 欧美在线网站| 少妇精品在线| 久久精品一区二区国产| 亚洲福利免费| 日韩精品一区二区三区av| 欧美~级网站不卡| 国产麻豆久久| 精品国产成人| 国产精品s色| 久久国产乱子精品免费女| 日本免费一区二区三区四区| 国产欧美69| 美女国产精品久久久| 日产欧产美韩系列久久99| 亚洲一区导航| 日韩激情啪啪| 国产欧美日韩亚洲一区二区三区| 日本va欧美va精品发布| 国产精品亚洲欧美日韩一区在线| 欧美日韩99| 视频在线不卡免费观看| 香蕉成人av| 亚洲激情另类| 欧美日韩调教| 免费久久精品| 欧美特黄一级| 日本一区中文字幕|