Redis第18講——Redis和Redission實現延遲消息

即使不是做電商業務的同學,也一定知道訂單超時關閉這種業務場景,這個場景大致就是用戶下單后,如果在一定時間內未支付(比如15分鐘、半小時),那么系統就會把這筆訂單給關閉掉。這個功能實現的方式有很多種,比如JDK中自帶的DelayQueue延遲隊列、Timer、ScheduledThreadPoolExecutor,強烈推薦的RocketMQ、RabblitMQ及Kafka等消息隊列,還有就是Hutool的SystemTimer、Netty的HashedWheelTimer等等,感興趣的可以去了解一下今天我們就先看看Redis和Redisson是如何實現延遲消息的。

一、Redis如何實現延遲消息

1.1 過期key監聽

很多人都知道Redis有一個過期監聽的功能,在redis.conf中加一條notify-keyspace-events開啟過期監聽,然后在代碼中實現KeyExpirationMessageListener就可以監聽key的過期消息了。

這樣就可以在接收到過期消息的時候進行關單操作了,但是這個方案并不推薦,Redis官方明確說過Redis并不保證key在過期的時候就能被立即刪除,更不保證這個消息能被立即發出,所以,消息延遲是必然的,數據量越大延遲的時間越長。

而且,在Redis5.0之前,這個消息是通過PUB/SUB模式發出的,不會做持久化,至于你有沒有接收到,有沒有消費成功,它不管,所以,如果發消息的時候客戶端掛了,之后再恢復的話,這個消息也就徹底丟了。

1.2 Zset

我們可以借助Redis中的有序集合——zset來實現這個功能,zset是一個有序集合,每一個元素(member)都關聯了一個score,可以通過score排序來取集合中的值。

我們可以將訂單超時時間的時間戳(下單時間+超時時間)作為score訂單號orderId作為成員(member),這樣redis會對zset按照score進行排序,再通過定時任務獲取“當前時間>=score"的延遲任務,獲取到之后就可以根據訂單號(member)進行關單操作。

這么實現的優點就是可以借助redis持久化和高可用機制,避免數據丟失。

1.3 zset實現超時關單Demo

步驟:

  • 用戶下單后,將信息寫入redis zset緩存中,score為當前時間+延遲時間的時間戳,member為訂單號。

  • 使用ZRangeByScore和WithScores命令,獲取當前時間戳之前的所有任務,并通過score判斷哪些任務已到期,進行關單處理。

  • 啟動一個額外的定時任務周期性檢查并處理已到期的訂單。

導入依賴:

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

Demo:

@Service
public class OrderDelayService {@Autowiredprivate RedisTemplate<String,String> redisTemplate;//延遲15分鐘private final long delayTime = 15*60*1000;//訂單號key前綴private static final String ORDER_KEY="order:";//延遲關單keyprivate static final String DELAY_KEY="close_orders";/*** 創建訂單* @param orderId*/public void createOrder(String orderId){//...//創建訂單成功//1.獲取當前時間戳long currentTime = System.currentTimeMillis();//2.score:當前時間+延遲時間long score=currentTime+delayTime;//3.加入redis zset集合redisTemplate.opsForZSet().add(DELAY_KEY, ?//redis keyORDER_KEY+orderId, //memberscore //score);}
?/*** 任務間隔一秒執行一次*/@Scheduled(fixedDelay = 1000)public void closeExpiredOrders(){//當前時間戳long currentTime = System.currentTimeMillis();ZSetOperations<String, String> zSetOps = redisTemplate.opsForZSet();//取出所有數據(已排好序)Set<String> orderKeys = zSetOps.range(DELAY_KEY, 0, -1);for (String orderKey : orderKeys) {//獲取score,double score = zSetOps.score(DELAY_KEY, orderKey);if(currentTime>=score){String orderId = orderKey.substring(ORDER_KEY.length());//進行關單操作...closeOrder(orderId)//從zset里移除該訂單zSetOps.remove(DELAY_KEY,orderKey);}}}
}

二、Redission如何實現延遲消息

2.1 實現原理

Redission中定義了分布式延遲隊列DelayedQueue,其實就是在zset的基礎上增加了一個基于內存的延遲隊列。

大致的流程如下:

當我們要添加一個數據到延遲隊列的時候,redission會把數據+超時時間放到zset中,并且起一個延時任務,當任務到期的時候,再去zset中把數據取出來,返回給客戶端使用。

2.2 案例

導入依賴:

 <dependency><groupId>org.redisson</groupId><artifactId>redisson-spring-boot-starter</artifactId><version>3.13.1</version>
</dependency>

定義一個客戶端:

@Component
public class RedissionConfig {@Bean(destroyMethod = "shutdown")public RedissonClient redssion(){Config config=new Config();config.useSingleServer().setAddress("redis://127.0.0.1");return Redisson.create(config);}
}

實現:

@Component
public class RedissionDelay {
?@Autowiredprivate RedissonClient client;/*** 創建訂單,并設置過期時間* @param orderId*/public void createOrder(String orderId){//...//創建訂單成功RBlockingDeque<Object> blockingDeque = client.getBlockingDeque("orderQueue");RDelayedQueue<Object> delayedQueue = client.getDelayedQueue(blockingDeque);//將訂單加入延遲隊列,延遲時間為15分鐘delayedQueue.offer(orderId,15, TimeUnit.MINUTES);}
?/*** 關單操作*/public void closeOrder(){RBlockingDeque<Object> orderQueue = client.getBlockingDeque("orderQueue");new Thread(() -> {while (true){try {String orderId = (String) orderQueue.take();//進行關單操作,closeOrder(orderId)} catch (InterruptedException e) {e.printStackTrace();}}});}
}

上述例子,我們用RDelayedQueue的offer方法將訂單添加到了延遲隊列,并指定了延遲時間,當元素的延遲時間到達時,Redission會將元素從RDelayedQueue轉移到與之關聯的RBlockingDeque。

然后在檢查是否要關單的時候,另起了一個線程,不斷循環讀取到期的訂單。值得注意的是 take方法從RBlockingDeque中獲取元素,這是一個阻塞操作,如果沒有元素,它會一直等到,直到有元素。

End:希望對大家有所幫助,如果有紕漏或者更好的想法,請您一定不要吝嗇你的賜教🙋。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/15925.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/15925.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/15925.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

unity開發Hololens 制作滑動框

一定要做到最后一步&#xff0c;才會有效果 1、創建空物體 ,并添加組件 創建空物體 命名ScrollingObjectCollection&#xff0c; 添加組件如下圖 下面是各個組件展開的內容 2、在ScrollingObjectCollection 下面創建兩個空物體&#xff0c;分別命名Container、Clipping…

運籌說 第115期 | 排隊論經典例題講解

通過前幾期的學習&#xff0c;我們已經學會了排隊論的基本概念、生滅過程和Poisson過程&#xff0c;等待制排隊模型、混合制排隊模型、其他排隊模型以及排隊系統優的定義與相關求解方法。在實際工作中&#xff0c;我們能發現排隊論在經濟管理中有著許多應用&#xff0c;本期小編…

大數據量上傳FTP

背景 筆者有一個需求是把將近一億條數據上傳到FTP服務器中&#xff0c;這些數據目前是存儲在mysql中&#xff0c;是通過關聯幾張表查詢出來的&#xff0c;查詢出來的數據結果集一共是6個字段。要求傳輸的時候拆分成一個個小文件&#xff0c;每個文件大小不能超過500M。我的測試…

FuTalk設計周刊-Vol.052

#AI漫談 熱點捕手 1.ChatGPT 大更新&#xff01;GPT-4 開始又變聰明了 OpenAI 官方宣布&#xff0c;新版 GPT-4 Turbo 今天開始向所有付費 ChatGPT 用戶開放。 鏈接https://www.pconline.com.cn/focus/1733/17330089.html 2.刷爆多模態任務榜單&#xff01;賈佳亞團隊Mini-G…

Linux下環境變量配置出錯導致基礎命令使用不了的問題解決

問題&#xff1a; 當配置環境變量&#xff1a; echo export PATH/home/ubuntu/.local/lib/python3.8/site-packages :$PATH >> ~/.bashrc 執行生效命令 source ~/.bashrc 出現所有的基礎操作命令&#xff1a;ls vim都使用不了 解決方式&#xff1a; 1&#xff09…

21.2zabbix低級自動發現-mysql多實例

配置mysql多實例 注釋&#xff1a;自動發現&#xff1a;創建監控主機&#xff1b;低級自動發現&#xff1a;創建監控項 mysql單實例是直接yum安裝&#xff0c;開啟mysql多實例 準備配置文件 #mysql3307實例 cp /etc/my.cnf /etc/my3307.cnf vim /etc/my3307.cnf [mysqld] dat…

lazarus-IDE 可以開發 Node.js 嗎?

Lazarus IDE 本身不是用來開發 Node.js 應用程序的工具&#xff0c;因為它是一個用于開發跨平臺應用程序的環境&#xff0c;類似于 C Builder 或 Delphi。Node.js 是一個基于 JavaScript 的運行時環境&#xff0c;通常使用 V8 引擎&#xff0c;用于構建異步、事件驅動的服務器端…

產品經理-流程圖結構圖(四)

1. 流程圖 1.1 概念 為了達到特定的目標而進行的一系列有邏輯性的操作步驟&#xff0c;由兩個及以上的步驟&#xff0c;完成一個完整的行為的過程&#xff0c;可稱之為流程 1.2 產品經理為什么需要繪制流程圖&#xff1f; 保證產品的使用邏輯合理順暢向項目組其他成員清晰的…

代碼隨想錄算法訓練營Day4|24. 兩兩交換鏈表中的節點、19.刪除鏈表的倒數第N個節點、 142.環形鏈表II、面試題 02.07. 鏈表相交

24. 兩兩交換鏈表中的節點 這道題的關鍵在于: 1、在置換兩個節點的時候&#xff0c;當前節點需要在這倆節點之前一個節點。并且要提前保存cur.next以及cur.next.next。 2、每次置換完一組節點&#xff0c;cur cur.next.next 3、判斷結束的標志&#xff1a;奇數個節點&#xf…

如何禁止U盤拷貝文件|禁止U盤使用的軟件有哪些

禁止U盤拷貝文件的方法有很多&#xff0c;比如使用注冊表、組策略編輯器等&#xff0c;但這些方法都適合個人&#xff0c;不適合企業&#xff0c;因為企業需要對下屬多臺電腦進行遠程管控&#xff0c;需要方便、省時、省力的方法。目前來說&#xff0c;最好的方法就是使用第三方…

Unity websocket客戶端

&#x1f3c6; 個人愚見&#xff0c;沒事寫寫筆記 &#x1f3c6;《博客內容》&#xff1a;Unity3D開發內容 &#x1f3c6;&#x1f389;歡迎 &#x1f44d;點贊?評論?收藏 &#x1f50e;目標&#xff1a;服務器和客戶端可以實時的傳輸信息 ??實現目標&#xff1a; 使用的w…

技術速遞|無障礙應用程序之旅:鍵盤可訪問性和 .NET MAUI

作者&#xff1a;Rachel Kang 排版&#xff1a;Alan Wang 首先讓我們一起來看看您的應用程序是否支持鍵盤訪問&#xff1a; 啟動您的其中一個應用。如果您的設備尚未連接物理鍵盤&#xff0c;請連接物理鍵盤。像平常一樣導航您的應用程序&#xff0c;并且僅使用鍵盤來執行此操…

如何使用Rust構建Python原生庫?注意,不是動態鏈接庫!!!

參考文檔&#xff1a;https://github.com/PyO3/pyo3 創建python虛擬環境&#xff1a; conda create --name pyo3 python3.11.7激活虛擬環境&#xff1a; conda activate pyo3安裝依賴&#xff1a; pip install maturin初始化項目&#xff1a; maturin init構建項目&#x…

設計模式--目錄

設計模式是軟件工程中為解決常見問題而總結出來的一系列通用解決方案。它們可以分為三大類別&#xff1a;創建型模式、結構型模式和行為型模式。下面列舉了一些常見的設計模式及其分類&#xff1a; 創建型模式(Creational Patterns) 創建型模式關注對象的創建過程&#xff0c…

小程序checkbox改成圓形與radio樣式保持一致

修改前 修改后 html: <view class"agreement"><checkbox value"{{ isAgreed }}" bind:tap"toggleCheckbox" /><text>我同意室外智能健身房 <text class"link" bind:tap"showUserProtocol">用戶協…

【JTS Topology Suite】Java對二維幾何進行平移、縮放、旋轉等坐標變換

JTS介紹 Github項目地址&#xff1a;https://github.com/locationtech/jts Maven庫地址&#xff1a;https://mvnrepository.com/artifact/org.locationtech.jts JTS Topology Suite是一個用于創建和操作二維矢量幾何的Java庫。 JTS有對應的.NET版本NetTopologySuite庫&…

P3128 [USACO15DEC] Max Flow P題解(樹上差分,最近公共祖先,圖論)

前言&#xff1a; 題目鏈接&#xff1a;P3128 [USACO15DEC] Max Flow P - 洛谷 | 計算機科學教育新生態 (luogu.com.cn) 講解&#xff1a; 這一題含金量真算高的&#xff0c;包含了建樹&#xff08;用了圖論的知識&#xff09;&#xff0c;求最近公共祖先&#xff08;倍增法…

2024目前網上最火短劇機器人做法,自動搜索發劇 自動更新資源 自動分享資源

目前整個項目圈子很多的短劇機器人&#xff0c;我寫的&#xff0c;自動搜索發劇&#xff0c;自動更新資源&#xff0c;自動分享資源&#xff0c;前段時間大部分做短劇的都是做的短劇分成&#xff0c;我的一個學員做的30W播放量才200塊收益&#xff0c;備受啟發&#xff0c;我就…

springboot社區助老志愿服務系統-計算機畢業設計源碼96682

摘要 大數據時代下&#xff0c;數據呈爆炸式地增長。為了迎合信息化時代的潮流和信息化安全的要求&#xff0c;利用互聯網服務于其他行業&#xff0c;促進生產&#xff0c;已經是成為一種勢不可擋的趨勢。在圖書館管理的要求下&#xff0c;開發一款整體式結構的社區助老志愿服務…

社交媒體數據恢復:綠洲

本教程將向您展示如何在綠洲平臺上備份和恢復數據&#xff0c;但不涉及推薦任何具體的數據恢復軟件。 一、綠洲平臺數據備份 為了確保數據的安全&#xff0c;在日常使用過程中&#xff0c;我們需要定期備份綠洲平臺上的數據。以下是備份綠洲平臺數據的步驟&#xff1a; 登錄綠…