開心一刻
我找了個女朋友,挺醜的那一種,她也知道自己丑,平常都不好意思和我一塊出門
昨晚,我帶她逛超市,聽到有兩個人在我們背後小聲嘀咕:“看我們前面,想不到這麼醜都有人要。”
女朋友聽後,羞的滿臉通紅,我想女朋友雖然醜但是對我很好,我不會嫌棄她的
後面兩個人繼續嘀咕:“是啊,那男人真醜!”
臥槽,小丑竟是我自己!
寫在前面
Redis 客戶端
除了 Redis 自己提供的命令列工具:redis-cli,還有各種針對不同程式語言的客戶端:Clients
Java 語言的 Redis 客戶端有很多,推薦使用的有:Jedis、lettuce、Redisson,而 Redisson 就是本文的主角之一
環境準備
Redis 版本:3.2.8
Redisson 版本:3.13.6
下文都是基於這兩個版本來進行講解的;不同的版本,功能、特性還是有所不同的,這點還是需要注意的
Redis 的釋出/訂閱
官方文件:Redis Pub/Sub
什麼是釋出/訂閱
Redis 提供了基於 “釋出 / 訂閱” 模式的訊息機制,此種模式下,訊息釋出者和訂閱者不進行直接通訊,釋出者向指定的頻道釋出訊息,訂閱該頻道的每個客戶端都可以收到該訊息
釋出訂閱模型如下:
四個角色:釋出者(Pub)、訂閱者(Sub)、對兩者解耦的中間方(Channel)、訊息(Message)
Sub 訂閱 Channel,Pub 向 Channel 釋出訊息(Message),Sub 就能收到 Pub 釋出的訊息了
以公眾號為例,我們(Sub)訂閱某個公眾號(Channel),公眾號作者(Pub)在公眾號每發表一篇文章(Message),就會向我們推送這篇文章,我們就可以瀏覽這篇文章了
當我們取消訂閱了,它就不會再向我們推送這篇文章了;只要這個公眾號一直在執行,就會一直有人訂閱它或者取消訂閱
可以將釋出/訂閱理解成分散式版的觀察者模式,關於觀察者模式,大家可以檢視:設計模式之觀察者模式 → 事件機制的底層原理
很多的 MQ 產品中都存在釋出/訂閱模式,只是各自的實現有細微差別
Redis 中釋出/訂閱相關的命令只有 6 個,我們在 redis-cli 下一個一個來看
SUBSCRIBE
通過該命令,客戶端可以訂閱一個或多個頻道
基本語法: subscribe channel [channel ...]
假設我們訂閱頻道:channel:1,可以如下操作
關於訂閱命令(subscribe、psubscribe)有兩點需要注意:
1、客戶端在執行訂閱命令後進入了訂閱狀態,只能接收 subscribe、psubscribe、unsubscribe、punsubscribe 這四個命令
在 redis-cli 下更是表現為阻塞狀態,只能接收訊息,不能輸入任何命令
但是我們要明白,redis 客戶端除了 redis-cli,還很多針對不同程式語言的客戶端
實際應用中,redis-cli 用的非常少,用的多的還是各種程式語言的 Redis 客戶端
2、新開啟的訂閱客戶端,無法接收到該頻道之前的訊息,因為 Redis 不會持久化釋出的訊息
PUBLISH
通過該命令,客戶端可以向某個頻道釋出一條訊息
基本語法: publish channel message
假設我們向頻道:channel:1 釋出訊息,可以如下操作
返回值: (integer) 1 表示有 1 個訂閱者收到了訊息
我們再看看之前的訂閱客戶端,收到了釋出的訊息
UNSUBSCRIBE
通過此命令,客戶端可以取消對指定頻道的訂閱,取消成功後不再接收該頻道釋出的訊息
基本語法: unsubscribe [channel [channel ...]]
我們取消對頻道:channel:1 的訂閱,可以如下操作
PSUBSCRIBE
按照模式訂閱,可以理解成正則匹配訂閱
subscribe 只能訂閱一個或多個具體的頻道,不能按正則匹配訂閱,而此命令正好彌補這個空缺
基本語法: psubscribe pattern [pattern ...]
我們訂閱以 channel:u 開頭的所有頻道,可以如下操作
此時,我們向頻道:channel:user 釋出訊息,那麼此客戶端也能收到訊息
PUNSUBSCRIBE
按照模式取消訂閱,可以理解成正則匹配取消訂閱
unsubscribe 只能對一個或多個具體的頻道取消訂閱,不能按正則匹配來取消訂閱,而此命令正好彌補這個空缺
基本語法: punsubscribe [pattern [pattern ...]]
我們對 channel:r 開頭的所有頻道取消訂閱,可以如下操作
我們可以將 psubscribe、punsubscribe 與 subscribe、unsubscribe 進行類比,便於理解
PUBSUB
該命令用於檢視訂閱與釋出系統狀態,它由數個不同格式的子命令組成
基本語法: pubsub subcommand [argument [argument ...]]
該命令用法比較靈活,常用的功能有如下幾個
1、檢視活躍的頻道
活躍的頻道指的是當前頻道至少有一個訂閱者
基本語法: pubsub channels [pattern] ,其中 [pattern] 是可以指定具體的模式
檢視所有活躍的頻道,可以如下操作
檢視符合某種模式的活躍頻道,可以如下操作
2、檢視頻道訂閱數
基本語法: pubsub numsub [channel ...]
channel:1 頻道的訂閱數是 1,channel:user 頻道的訂閱數也是 1
3、檢視模式訂閱數
基本語法: pubsub numpat
返回的不是訂閱模式的客戶端的數量, 而是客戶端訂閱的所有模式的數量總和
Redisson 釋出/訂閱
上面講了那麼多,其實都是在 redis-cli 下自嗨,如何在實際專案中應用起來了,我們基於 Redisson 來實現個簡單示例
訂閱端
釋出端
完整程式碼:pubsub,執行結果如下
至此,相信大家對 Redis 的釋出/訂閱有了一定的瞭解了
Redis 的 Lua
官方文件:Redis Lua scripting
關於 Lua,本文不作詳細介紹;語法比較簡單,基本都能看懂,感興趣的可以去看它的官方文件:Lua Documentation
Redis 提供了一系列的命令供我們使用:Redis Commands,基本上能滿足我們的絕大部分需求
但是,總有一些特殊的需求遊離在三界之外,不在五行之中,不能通過其中的某個命令直接實現
有人可能就會說了:一個命令不行,那就多個命令組合實現嘛
但是,我們需要考慮到:多個命令組合能保證原子性嗎,如果有邏輯處理又該怎麼辦?
Redis 早已替我們想好了解決辦法,那就是:Lua 指令碼
在 Redis 中執行 Lua 指令碼有兩種方法:eval 和 evalsha
eval
基本語法: eval script numkeys key [key ...] arg [arg ...]
其中 script 表示 Lua 指令碼,numkeys 表示 key 個數
通過一個具體案例,我們就能理解了
其中表示 .. 表示連線兩個字串
如果 Lua 指令碼太長,還可以使用 redis-cli --eval 直接執行檔案
基本語法: redis-cli --eval script key [key...] , arg [arg ...]
注意:key 與 arg 之間是 , ,英文逗號前後都有一個空格
hello.lua 檔案內容: return 'hello '..KEYS[1]..ARGV[1]
evalsha
除了 eval,Redis 還提供了 evalsha 來執行 Lua 指令碼
基本語法: evalsha sha1 numkeys key [key ...] arg [arg ...]
使用 evalsha 之前需要將 Lua 指令碼載入到 Redis 服務端,得到該指令碼的 SHA1 校驗和,然後將 SHA1 作為 evalsha 的入參執行對應的 Lua 指令碼
指令碼會常駐 Redis 服務端,客戶端執行指令碼時不需要每次都傳遞指令碼到服務端,使得指令碼得以複用,降低了引數傳遞的開銷
載入指令碼基本語法: redis-cli script load script
得到 SHA1: 5a8bcaa0ac71ab25ea5c504d61964859fffc20ce ,再執行 evalsha 命令
Lua 的 Redis API
Lua 可以使用 redis.call 函式實現對 Redis 命令的呼叫,例如:
另外還可以使用 redis.pcall 函式實現對 Redis 命令的呼叫
redis.call 和 redis.pcall 的區別在於,如果 redis.call 執行失敗,那麼指令碼執行結束會直接返回錯誤,而 redis.pcall 會忽略錯誤繼續執行指令碼
Lua 帶來的好處
Lua 為 Redis 開發和運維人員帶來了如下三個好處:
1、Lua 指令碼在 Redis 中是原子執行的,執行過程中不會插入其他命令
2、通過 Lua 指令碼,我可以創造出自己定製的命令,並可以將這些命令常駐在記憶體,實現複用
3、Lua 指令碼可以將多條命令一次性打包,有效減少網路開銷
Redisson Lua
基於 Redisson,我們來看看 Lua 的簡單使用
完整程式碼:LuaDemo,執行結果如下:
LuaDemo.java 中有個方法 distLockTest ,有興趣的可以看看,對理解 Redisson 分散式鎖的實現有幫助
細節疑問
給大家留兩個問題
1、客戶端未主動取消訂閱,而是直接斷開連線,Redis 服務端會如何處理該客戶端訂閱的那些頻道
2、lua 指令碼保證的是執行該指令碼的過程中,不能有其他命令插入,但是如果指令碼中的某個命令出錯了,Redis 會如何處理
總結
1、Redis 釋出訂閱模式可以類比觀察者模式,便於理解
涉及 4 個角色,理清楚它們各自的作用就好理解了
2、Lua 在 Redis 中非常靈活,相當於給我們留了一個自定義命令的介面
3、Redis 客戶端有很多,我們不能只侷限於 redis-cli
參考
《Redis開發與運維》