主頁 > 後端開發 > Redisson 分布式鎖實作之前置篇 → Redis 的發布/訂閱 與 Lua

Redisson 分布式鎖實作之前置篇 → Redis 的發布/訂閱 與 Lua

2021-06-16 09:46:31 後端開發

開心一刻

  我找了個女朋友,挺丑的那一種,她也知道自己丑,平常都不好意思和我一塊出門

  昨晚,我帶她逛超市,聽到有兩個人在我們背后小聲嘀咕:“看咱前面,想不到這么丑都有人要,”

  女朋友聽后,羞的滿臉通紅,我想女朋友雖然丑但是對我很好,我不會嫌棄她的

  后面兩個人繼續嘀咕:“是啊,那男人真丑!”

  臥槽,小丑竟是我自己!

寫在前面

  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開發與運維》

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/287725.html

標籤:Java

上一篇:SpringBoot2 引數管理實踐,入參出參與校驗

下一篇:PHP中的PDO操作學習(二)預處理陳述句及事務

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more