欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時(shí)間:2021年05月11日 09:35:46   作者:酸酸的酸酸醬  
這篇文章主要介紹了Springboot+rabbitmq實(shí)現(xiàn)延時(shí)隊(duì)列的兩種方式,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

什么是延時(shí)隊(duì)列,延時(shí)隊(duì)列應(yīng)用于什么場(chǎng)景

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

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

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

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

利用TTL DLX實(shí)現(xiàn)延時(shí)隊(duì)列的方式

TTL DLX是什么

TTL

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

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

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

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

<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è)立即消費(fèi)隊(duì)列
    @Bean
    public Queue immediateQueue() {
        // 第一個(gè)參數(shù)是創(chuàng)建的queue的名字,第二個(gè)參數(shù)是是否支持持久化
        return new Queue(Constants.IMMEDIATE_QUEUE, true);
    }

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

    @Bean
    public DirectExchange deadLetterExchange() {
        // 一共有三種構(gòu)造方法,可以只傳exchange的名字, 第二種,可以傳exchange名字,是否支持持久化,是否可以自動(dòng)刪除,
        //第三種在第二種參數(shù)上可以增加Map,Map中可以存放自定義exchange中的參數(shù)
        return new DirectExchange(Constants.DEAD_LETTER_EXCHANGE, true, false);
    }

    @Bean
    //把立即消費(fèi)的隊(duì)列和立即消費(fèi)的exchange綁定在一起
    public Binding immediateBinding() {
        return BindingBuilder.bind(immediateQueue()).to(immediateExchange()).with(Constants.IMMEDIATE_ROUTING_KEY);
    }

    @Bean
    //把立即消費(fèi)的隊(duì)列和立即消費(fèi)的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;
        });
    }
}

消費(fèi)者消費(fèi)消息

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

測(cè)試類

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("預(yù)定房子");
        booking.setBookingTime(new Date());
        booking.setOperatorName("hellen");
     immediateSender.send(booking, 1000);
 }
}

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

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

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

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

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

x-delay-message安裝

介紹Ubuntu系統(tǒng)下插件安裝方式:
選擇rabbitmq_delayed_message_exchange插件,選擇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同理

Springboot集成rabbitmq實(shí)現(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;

@Configuration
public class XdelayConfig {

    // 創(chuàng)建一個(gè)立即消費(fèi)隊(duì)列
    @Bean
    public Queue immediateQueue() {
        // 第一個(gè)參數(shù)是創(chuàng)建的queue的名字,第二個(gè)參數(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;

@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("預(yù)定房子");
        booking.setBookingTime(new Date());
        booking.setOperatorName("hellen");
        xdelaySender.send(booking, 2000);
 }
}

到此這篇關(guān)于Springboot+rabbitmq實(shí)現(xiàn)延時(shí)隊(duì)列的兩種方式的文章就介紹到這了,更多相關(guān)Springboot rabbitmq延時(shí)隊(duì)列內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java使用阻塞隊(duì)列控制線程通信的方法實(shí)例詳解

    Java使用阻塞隊(duì)列控制線程通信的方法實(shí)例詳解

    這篇文章主要介紹了Java使用阻塞隊(duì)列控制線程通信的方法,結(jié)合實(shí)例形式詳細(xì)分析了java使用阻塞隊(duì)列控制線程通信的相關(guān)原理、方法及操作注意事項(xiàng),需要的朋友可以參考下
    2019-09-09
  • 解決javac不是內(nèi)部或外部命令,也不是可運(yùn)行程序的報(bào)錯(cuò)問題

    解決javac不是內(nèi)部或外部命令,也不是可運(yùn)行程序的報(bào)錯(cuò)問題

    在學(xué)著使用Java的命令行來編譯java文件的時(shí)候,遇到了這個(gè)問題,本文主要介紹了解決javac不是內(nèi)部或外部命令,也不是可運(yùn)行程序的報(bào)錯(cuò)問題,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-04-04
  • 如何在spring官網(wǎng)查找XML基礎(chǔ)配置文件

    如何在spring官網(wǎng)查找XML基礎(chǔ)配置文件

    這篇文章主要介紹了如何在spring官網(wǎng)查找XML基礎(chǔ)配置文件,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-10-10
  • 值得收藏的2017年Java開發(fā)崗位面試題

    值得收藏的2017年Java開發(fā)崗位面試題

    這篇文章主要為大家推薦一份值得收藏的2017年Java開發(fā)崗位面試題,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-08-08
  • 一文詳解Java中的可變對(duì)象(Mutable)與不可變對(duì)象(Immutable)

    一文詳解Java中的可變對(duì)象(Mutable)與不可變對(duì)象(Immutable)

    如何在 Java 中創(chuàng)建不可變對(duì)象?我以前以為所有對(duì)象都是不可變的,因?yàn)槿绻愀淖円粋€(gè) String 實(shí)例的內(nèi)容,它總是會(huì)創(chuàng)建一個(gè)新的 String 對(duì)象并指向該對(duì)象,在本文中,我不僅將分享在 Java 中Immutable的步驟,還將討論可變對(duì)象與不可變對(duì)象及其優(yōu)缺點(diǎn)
    2023-11-11
  • java銀行管理系統(tǒng)源碼

    java銀行管理系統(tǒng)源碼

    這篇文章主要為大家詳細(xì)介紹了java銀行管理系統(tǒng)源碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-12-12
  • 在IDEA中實(shí)現(xiàn)同時(shí)運(yùn)行2個(gè)相同的java程序

    在IDEA中實(shí)現(xiàn)同時(shí)運(yùn)行2個(gè)相同的java程序

    這篇文章主要介紹了在IDEA中實(shí)現(xiàn)同時(shí)運(yùn)行2個(gè)相同的java程序,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-02-02
  • HDFS?Balancer負(fù)載均衡器及語(yǔ)法詳解

    HDFS?Balancer負(fù)載均衡器及語(yǔ)法詳解

    這篇文章主要為大家介紹了HDFS?Balancer負(fù)載均衡器及語(yǔ)法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-03-03
  • Java8 Stream flatmap中間操作用法解析

    Java8 Stream flatmap中間操作用法解析

    這篇文章主要介紹了Java8 Stream flatmap中間操作用法解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-07-07
  • windows上nacos自啟動(dòng)的三種方法小結(jié)

    windows上nacos自啟動(dòng)的三種方法小結(jié)

    本文主要給大家介紹了windows上nacos自啟動(dòng)的三種方法,借助WinSW.exe添加到服務(wù)列表,修改nacos啟動(dòng)配置以及以開機(jī)"啟動(dòng)"方式——啟動(dòng)Nacos的startup.cmd這三種方法,文中通過圖文講解的非常詳細(xì),需要的朋友可以參考下
    2023-12-12

最新評(píng)論