电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

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

Springboot+rabbitmq實現(xiàn)延時隊列的兩種方式

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

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

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

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

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

利用TTL DLX實現(xiàn)延時隊列的方式TTL DLX是什么

TTL

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

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

Springboot集成rabbitmq實現(xiàn)第一種方式

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

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

為什么會出現(xiàn)這樣的現(xiàn)象呢?

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

利用Rabbitmq的插件x-delay-message實現(xiàn)

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

x-delay-message安裝

介紹Ubuntu系統(tǒng)下插件安裝方式:選擇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實現(xiàn)第二種方式

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 { // 創(chuàng)建一個立即消費隊列 @Bean public Queue immediateQueue() {// 第一個參數(shù)是創(chuàng)建的queue的名字,第二個參數(shù)是是否支持持久化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實現(xiàn)延時隊列的兩種方式的文章就介紹到這了,更多相關Springboot rabbitmq延時隊列內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持好吧啦網(wǎng)!

標簽: Spring
相關文章:
主站蜘蛛池模板: 金属检测机_金属分离器_检针验针机_食品药品金属检探测仪器-广东善安科技 | 一体化净水器_一体化净水设备_一体化水处理设备-江苏旭浩鑫环保科技有限公司 | 深圳工程师职称评定条件及流程_深圳职称评审_职称评审-职称网 | 金现代信息产业股份有限公司--数字化解决方案供应商 | 预制直埋蒸汽保温管-直埋管道-聚氨酯发泡保温管厂家 - 唐山市吉祥保温工贸有限公司 | 华中线缆有限公司-电缆厂|电缆厂家|电线电缆厂家| 硅胶管挤出机厂家_硅胶挤出机生产线_硅胶条挤出机_臣泽智能装备 贵州科比特-防雷公司厂家提供贵州防雷工程,防雷检测,防雷接地,防雷设备价格,防雷产品报价服务-贵州防雷检测公司 | 高防护蠕动泵-多通道灌装系统-高防护蠕动泵-www.bjhuiyufluid.com慧宇伟业(北京)流体设备有限公司 | 不锈钢反应釜,不锈钢反应釜厂家-价格-威海鑫泰化工机械有限公司 不干胶标签-不干胶贴纸-不干胶标签定制-不干胶标签印刷厂-弗雷曼纸业(苏州)有限公司 | 包塑软管|金属软管|包塑金属软管-闵彬管业 | 贵州科比特-防雷公司厂家提供贵州防雷工程,防雷检测,防雷接地,防雷设备价格,防雷产品报价服务-贵州防雷检测公司 | 上海三信|ph计|酸度计|电导率仪-艾科仪器 | 烟台金蝶财务软件,烟台网站建设,烟台网络推广 | 不锈钢轴流风机,不锈钢电机-许昌光维防爆电机有限公司(原许昌光维特种电机技术有限公司) | 丽陂特官网_手机信号屏蔽器_Wifi信号干扰器厂家_学校考场工厂会议室屏蔽仪 | 上海噪音治理公司-专业隔音降噪公司-中广通环保 | 东莞市踏板石餐饮管理有限公司_正宗桂林米粉_正宗桂林米粉加盟_桂林米粉加盟费-东莞市棒子桂林米粉 | 珠光砂保温板-一体化保温板-有釉面发泡陶瓷保温板-杭州一体化建筑材料 | 塑料瓶罐_食品塑料瓶_保健品塑料瓶_调味品塑料瓶–东莞市富慷塑料制品有限公司 | 光栅尺厂家_数显表维修-苏州泽升精密机械 | 没斑啦-专业的祛斑美白嫩肤知识网站-去斑经验分享 | 优宝-汽车润滑脂-轴承润滑脂-高温齿轮润滑油脂厂家 | 生态板-实木生态板-生态板厂家-源木原作生态板品牌-深圳市方舟木业有限公司 | 广州中央空调回收,二手中央空调回收,旧空调回收,制冷设备回收,冷气机组回收公司-广州益夫制冷设备回收公司 | 浙江富广阀门有限公司| 篮球地板厂家_舞台木地板品牌_体育运动地板厂家_凯洁地板 | pos机办理,智能/扫码/二维码/微信支付宝pos机-北京万汇通宝商贸有限公司 | 合肥卓创建筑装饰,专业办公室装饰、商业空间装修与设计。 | 充气膜专家-气膜馆-PTFE膜结构-ETFE膜结构-商业街膜结构-奥克金鼎 | 电子厂招聘_工厂招聘_普工招聘_小时工招聘信息平台-众立方招工网 | 合肥升降机-合肥升降货梯-安徽升降平台「厂家直销」-安徽鼎升自动化科技有限公司 | 解放卡车|出口|济南重汽|报价大全|山东三维商贸有限公司 | 振动时效_振动时效仪_超声波冲击设备-济南驰奥机电设备有限公司 北京宣传片拍摄_产品宣传片拍摄_宣传片制作公司-现像传媒 | 碳纤维复合材料制品生产定制工厂订制厂家-凯夫拉凯芙拉碳纤维手机壳套-碳纤维雪茄盒外壳套-深圳市润大世纪新材料科技有限公司 | 储能预警-储能消防系统-电池舱自动灭火装置-四川千页科技股份有限公司官网 | 深圳工程师职称评定条件及流程_深圳职称评审_职称评审-职称网 | 工业车间焊接-整体|集中除尘设备-激光|等离子切割机配套除尘-粉尘烟尘净化治理厂家-山东美蓝环保科技有限公司 | 热镀锌槽钢|角钢|工字钢|圆钢|H型钢|扁钢|花纹板-天津千百顺钢铁贸易有限公司 | 档案密集柜_手动密集柜_智能密集柜_内蒙古档案密集柜-盛隆柜业内蒙古密集柜直销中心 | ERP企业管理系统永久免费版_在线ERP系统_OA办公_云版软件官网 | 贴片电感_贴片功率电感_贴片绕线电感_深圳市百斯特电子有限公司 贴片电容代理-三星电容-村田电容-风华电容-国巨电容-深圳市昂洋科技有限公司 |