Redis的發(fā)布與訂閱

要知道redis是有消息的發(fā)布和訂閱功能的,我們可以利用它的發(fā)布和訂閱功能非常簡單地實(shí)現(xiàn)一些比較實(shí)用的功能穆刻。

打個(gè)比方秉扑,如何實(shí)現(xiàn)自動(dòng)關(guān)閉超時(shí)未支付的訂單吸占?

我們通常的做法是寫一個(gè)定時(shí)器巧鸭,定時(shí)去掃描未支付的訂單瓶您,當(dāng)發(fā)現(xiàn)下單時(shí)間超過我們?cè)O(shè)置的閾值時(shí)就去關(guān)閉訂單麻捻。 這樣做有一個(gè)問題:訂單可能會(huì)延時(shí)關(guān)閉纲仍,假如設(shè)置5分鐘掃描一次未支付訂單,未支付訂單有效時(shí)間是15分鐘贸毕,那么就有可能一些訂單到了(15+5)分鐘-1秒才會(huì)被關(guān)閉郑叠。

那么這個(gè)時(shí)候使用redis發(fā)布訂閱功能就非常方便了,我們只需要在用戶下訂單的時(shí)候把訂單號(hào)作為key寫入redis明棍,并設(shè)置一個(gè)15分鐘的有效期乡革。然后訂閱這個(gè)key的過期事件,如果用戶在15分鐘之內(nèi)支付了訂單我們就直接刪除這個(gè)key。如果到了15分鐘key自動(dòng)過期了沸版,我們就會(huì)接收到redis的消息通知嘁傀,這個(gè)時(shí)候就可以直接關(guān)閉訂單了。

開啟redis消息訂閱

打開redis.config配置文件视粮,搜索notify-keyspace-events就會(huì)看到redis發(fā)布訂閱的配置:

# It is possible to select the events that Redis will notify among a set
# of classes. Every class is identified by a single character:
#
#  K     Keyspace events, published with __keyspace@<db>__ prefix.
#  E     Keyevent events, published with __keyevent@<db>__ prefix.
#  g     Generic commands (non-type specific) like DEL, EXPIRE, RENAME, ...
#  $     String commands
#  l     List commands
#  s     Set commands
#  h     Hash commands
#  z     Sorted set commands
#  x     Expired events (events generated every time a key expires)
#  e     Evicted events (events generated when a key is evicted for maxmemory)
#  A     Alias for g$lshzxe, so that the "AKE" string means all the events.
#
#  The "notify-keyspace-events" takes as argument a string that is composed
#  of zero or multiple characters. The empty string means that notifications
#  are disabled.
#
#  Example: to enable list and generic events, from the point of view of the
#           event name, use:
#
#  notify-keyspace-events Elg
#
#  Example 2: to get the stream of the expired keys subscribing to channel
#             name __keyevent@0__:expired use:
#
#  notify-keyspace-events Ex

redis接收事件類型一共有兩種细办,keyspacekeyeventkeyspace是key觸發(fā)的具體操作蕾殴,keyevent為操作影響的鍵名笑撞。g,$,l,s,h,z,x,e,A表示監(jiān)聽什么樣的事件。

舉個(gè)例子我們配置訂閱類型為KEx钓觉,我們就可以接收到兩種key過期后產(chǎn)生的消息茴肥。

notify-keyspace-events "KEx" #設(shè)置監(jiān)聽類型

重啟redis。

開始監(jiān)聽redis消息訂閱

開啟三個(gè)客戶端:
訂閱redis key為kname的事件

127.0.0.1:6379> subscribe __keyspace@0__:kname
Reading messages... (press Ctrl-C to quit)
1) "subscribe"
2) "__keyspace@0__:kname"
3) (integer) 1

訂閱redis key 的過期事件

127.0.0.1:6379> subscribe __keyevent@0__:expired
Reading messages... (press Ctrl-C to quit)
1) "subscribe"
2) "__keyevent@0__:expired"
3) (integer) 1

設(shè)置kname為zhangsan過期時(shí)間為2秒

127.0.0.1:6379> set kname "zhansan" ex 2 

兩秒后訂閱的客戶端分別收到了redis的消息通知

127.0.0.1:6379> subscribe __keyspace@0__:kname
Reading messages... (press Ctrl-C to quit)
1) "subscribe"
2) "__keyspace@0__:kname"
3) (integer) 1
1) "message"
2) "__keyspace@0__:kname"
3) "expired"  #監(jiān)聽到kname過期了
127.0.0.1:6379> subscribe __keyevent@0__:expired
Reading messages... (press Ctrl-C to quit)
1) "subscribe"
2) "__keyevent@0__:expired"
3) (integer) 1
1) "message"
2) "__keyevent@0__:expired"
3) "kname"  #監(jiān)聽到有一個(gè)過期的key為kname

上面這個(gè)訂閱到一個(gè)redis庫的事件荡灾,要想訂閱所有的redis庫就需要使用通配符了瓤狐。

psubscribe __key*@*__:* # 這里注意通配的命令是p開頭
在springboot里如何訂閱redis事件

添加pom依賴

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

yaml文件配置

server:
  port: 8098
spring:
  application:
    name: redis-subscribe-service
  redis:
    host: 127.0.0.1
    port: 6379
    password: red123456

編寫訂閱service

package com.me.binf.service;

import org.springframework.stereotype.Service;

@Service
public class MessageReceiver {
    //接收消息的方法
    public void receiveMessage(String message){
        //message接收到的過期key
        System.out.println("Redis 監(jiān)聽到過期的key有:"+message);
    }
}

添加redis配置

package com.me.binf.config;

import com.icodingedu.supermall.service.MessageReceiver;
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.PatternTopic;
import org.springframework.data.redis.listener.RedisMessageListenerContainer;
import org.springframework.data.redis.listener.adapter.MessageListenerAdapter;

@Configuration
public class RedisMessageConfig {

    @Bean
    RedisMessageListenerContainer container(RedisConnectionFactory connectionFactory,
                                            MessageListenerAdapter listenerAdapter){
        RedisMessageListenerContainer container = new RedisMessageListenerContainer();
        container.setConnectionFactory(connectionFactory);
        //訂閱觸發(fā)的通道
        container.addMessageListener(listenerAdapter,
                new PatternTopic("__keyevent@0__:expired"));
        return container;
    }

    @Bean
    MessageListenerAdapter listenerAdapter(MessageReceiver receiver){
        return new MessageListenerAdapter(receiver,"receiveMessage");
    }
}

設(shè)置kname過期后接收到消息:

Redis 監(jiān)聽到過期的key有:kname
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市卧晓,隨后出現(xiàn)的幾起案子芬首,更是在濱河造成了極大的恐慌,老刑警劉巖逼裆,帶你破解...
    沈念sama閱讀 222,681評(píng)論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件郁稍,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡胜宇,警方通過查閱死者的電腦和手機(jī)耀怜,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,205評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來桐愉,“玉大人财破,你說我怎么就攤上這事〈踊澹” “怎么了左痢?”我有些...
    開封第一講書人閱讀 169,421評(píng)論 0 362
  • 文/不壞的土叔 我叫張陵,是天一觀的道長系洛。 經(jīng)常有香客問我俊性,道長,這世上最難降的妖魔是什么描扯? 我笑而不...
    開封第一講書人閱讀 60,114評(píng)論 1 300
  • 正文 為了忘掉前任定页,我火速辦了婚禮,結(jié)果婚禮上绽诚,老公的妹妹穿的比我還像新娘典徊。我一直安慰自己杭煎,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,116評(píng)論 6 398
  • 文/花漫 我一把揭開白布卒落。 她就那樣靜靜地躺著羡铲,像睡著了一般。 火紅的嫁衣襯著肌膚如雪儡毕。 梳的紋絲不亂的頭發(fā)上犀勒,一...
    開封第一講書人閱讀 52,713評(píng)論 1 312
  • 那天,我揣著相機(jī)與錄音妥曲,去河邊找鬼贾费。 笑死,一個(gè)胖子當(dāng)著我的面吹牛檐盟,可吹牛的內(nèi)容都是我干的褂萧。 我是一名探鬼主播,決...
    沈念sama閱讀 41,170評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼葵萎,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼导犹!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起羡忘,我...
    開封第一講書人閱讀 40,116評(píng)論 0 277
  • 序言:老撾萬榮一對(duì)情侶失蹤谎痢,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后卷雕,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體节猿,經(jīng)...
    沈念sama閱讀 46,651評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,714評(píng)論 3 342
  • 正文 我和宋清朗相戀三年漫雕,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了滨嘱。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,865評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡浸间,死狀恐怖太雨,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情魁蒜,我是刑警寧澤囊扳,帶...
    沈念sama閱讀 36,527評(píng)論 5 351
  • 正文 年R本政府宣布,位于F島的核電站兜看,受9級(jí)特大地震影響锥咸,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜铣减,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,211評(píng)論 3 336
  • 文/蒙蒙 一她君、第九天 我趴在偏房一處隱蔽的房頂上張望脚作。 院中可真熱鬧葫哗,春花似錦缔刹、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,699評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至捺典,卻和暖如春鸟廓,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背襟己。 一陣腳步聲響...
    開封第一講書人閱讀 33,814評(píng)論 1 274
  • 我被黑心中介騙來泰國打工引谜, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人擎浴。 一個(gè)月前我還...
    沈念sama閱讀 49,299評(píng)論 3 379
  • 正文 我出身青樓员咽,卻偏偏與公主長得像,于是被迫代替她去往敵國和親贮预。 傳聞我的和親對(duì)象是個(gè)殘疾皇子贝室,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,870評(píng)論 2 361