<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          Spring Boot 監(jiān)聽 Redis Key 失效事件實(shí)現(xiàn)定時任務(wù)

          共 2969字,需瀏覽 6分鐘

           ·

          2020-10-25 04:37

          來源:antoniopeng.com/Redis

          • 業(yè)務(wù)場景
          • 實(shí)現(xiàn)思路
          • 開啟 Redis key 過期提醒
          • 引入依賴
          • 相關(guān)配置

          業(yè)務(wù)場景

          我們以訂單功能為例說明下:

          生成訂單后一段時間不支付訂單會自動關(guān)閉。最簡單的想法是設(shè)置定時任務(wù)輪詢,但是每個訂單的創(chuàng)建時間不一樣,定時任務(wù)的規(guī)則無法設(shè)定,如果將定時任務(wù)執(zhí)行的間隔設(shè)置的過短,太影響效率。

          還有一種想法,在用戶進(jìn)入訂單界面的時候,判斷時間執(zhí)行相關(guān)操作。方式可能有很多,在這里介紹一種監(jiān)聽 Redis 鍵值對過期時間來實(shí)現(xiàn)訂單自動關(guān)閉。

          實(shí)現(xiàn)思路

          在生成訂單時,向 Redis 中增加一個 KV 鍵值對,K 為訂單號,保證通過 K 能定位到數(shù)據(jù)庫中的某個訂單即可,V 可為任意值。

          假設(shè),生成訂單時向 Redis 中存放 K 為訂單號,V 也為訂單號的鍵值對,并設(shè)置過期時間為 30 分鐘,如果該鍵值對在 30 分鐘過期后能夠發(fā)送給程序一個通知,或者執(zhí)行一個方法,那么即可解決訂單關(guān)閉問題。

          實(shí)現(xiàn):通過監(jiān)聽 Redis 提供的過期隊(duì)列來實(shí)現(xiàn),監(jiān)聽過期隊(duì)列后,如果 Redis 中某一個 KV 鍵值對過期了,那么將向監(jiān)聽者發(fā)送消息,監(jiān)聽者可以獲取到該鍵值對的 K,注意,是獲取不到 V 的,因?yàn)橐呀?jīng)過期了,這就是上面所提到的,為什么要保證能通過 K 來定位到訂單,而 V 為任意值即可。拿到 K 后,通過 K 定位訂單,并判斷其狀態(tài),如果是未支付,更新為關(guān)閉,或者取消狀態(tài)即可。

          開啟 Redis key 過期提醒

          修改 redis 相關(guān)事件配置。找到 redis 配置文件 redis.conf,查看 notify-keyspace-events 配置項(xiàng),如果沒有,添加 notify-keyspace-events Ex,如果有值,則追加 Ex,相關(guān)參數(shù)說明如下:

          • K:keyspace 事件,事件以 keyspace@ 為前綴進(jìn)行發(fā)布
          • E:keyevent 事件,事件以 keyevent@ 為前綴進(jìn)行發(fā)布
          • g:一般性的,非特定類型的命令,比如del,expire,rename等
          • $:字符串特定命令
          • l:列表特定命令
          • s:集合特定命令
          • h:哈希特定命令
          • z:有序集合特定命令
          • x:過期事件,當(dāng)某個鍵過期并刪除時會產(chǎn)生該事件
          • e:驅(qū)逐事件,當(dāng)某個鍵因 maxmemore 策略而被刪除時,產(chǎn)生該事件
          • A:g$lshzxe的別名,因此”AKE”意味著所有事件

          引入依賴

          在 pom.xml 中添加 org.springframework.boot:spring-boot-starter-data-redis 依賴


          ????org.springframework.boot
          ????spring-boot-starter-data-redis

          相關(guān)配置

          定義配置 RedisListenerConfig 實(shí)現(xiàn)監(jiān)聽 Redis key 過期時間

          import?org.springframework.context.annotation.Bean;
          import?org.springframework.context.annotation.Configuration;
          import?org.springframework.data.redis.connection.RedisConnectionFactory;
          import?org.springframework.data.redis.listener.RedisMessageListenerContainer;

          @Configuration
          public?class?RedisListenerConfig?{

          ????@Bean
          ????RedisMessageListenerContainer?container(RedisConnectionFactory?connectionFactory)?{

          ????????RedisMessageListenerContainer?container?=?new?RedisMessageListenerContainer();
          ????????container.setConnectionFactory(connectionFactory);
          ????????return?container;
          ????}
          }

          定義監(jiān)聽器 RedisKeyExpirationListener,實(shí)現(xiàn)KeyExpirationEventMessageListener?接口,查看源碼發(fā)現(xiàn),該接口監(jiān)聽所有 db 的過期事件?keyevent@*:expired"

          import?org.springframework.data.redis.connection.Message;
          import?org.springframework.data.redis.listener.KeyExpirationEventMessageListener;
          import?org.springframework.data.redis.listener.RedisMessageListenerContainer;
          import?org.springframework.stereotype.Component;

          /**
          ?*?監(jiān)聽所有db的過期事件__keyevent@*__:expired"
          ?*/

          @Component
          public?class?RedisKeyExpirationListener?extends?KeyExpirationEventMessageListener?{

          ????public?RedisKeyExpirationListener(RedisMessageListenerContainer?listenerContainer)?{
          ????????super(listenerContainer);
          ????}

          ????/**
          ?????*?針對?redis?數(shù)據(jù)失效事件,進(jìn)行數(shù)據(jù)處理
          ?????*?@param?message
          ?????*?@param?pattern
          ?????*/

          ????@Override
          ????public?void?onMessage(Message?message,?byte[]?pattern)?{

          ????????//?獲取到失效的?key,進(jìn)行取消訂單業(yè)務(wù)處理
          ????????String?expiredKey?=?message.toString();
          ????????System.out.println(expiredKey);
          ????}
          }


          往期推薦

          前瞻:在 Java 16 中會帶來哪些新特性?

          高可用 Prometheus 的常見問題

          Fastjson < 1.2.68版本反序列化漏洞分析篇

          音效摸魚還不夠爽?試試IDE里打幾盤魂斗羅?

          一個讓你敲代碼的同時,找回童年樂趣的 IntelliJ 插件


          掃一掃,關(guān)注我

          一起學(xué)習(xí),一起進(jìn)步

          每周贈書,福利不斷

          深度內(nèi)容

          推薦加入


          最近熱門內(nèi)容回顧? ?#技術(shù)人系列


          瀏覽 35
          點(diǎn)贊
          評論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點(diǎn)贊
          評論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  操逼视频在线观看 | 操逼小黄片 | 日本无码中文字幕综合一区 | 一级欧美视频 | 日韩w w w x x x |