Redis的“假事務”與分散式鎖

来源:https://www.cnblogs.com/yulinfeng/archive/2020/02/25/12363960.html
-Advertisement-
Play Games

關註公眾號:CoderBuff,回覆“redis”獲取《Redis5.x入門教程》完整版PDF。 《Redis5.x入門教程》目錄 "第一章 · 準備工作" "第二章 · 數據類型" "第三章 · ​命令" "第四章 ​· 配置" "第五章 · Java客戶端(上)" "第六章 · 事務" "第七章 ...


關註公眾號:CoderBuff,回覆“redis”獲取《Redis5.x入門教程》完整版PDF。

第六章 · 事務

我們在學習MySQL的存儲殷勤時知道,MySQL中innodb支持事務而myisam不支持事務。而事務具有四個特性:

  • 一致性
  • 原子性
  • 隔離性
  • 持久性

在redis儘管提供了事務相關的命令,但實際上它是一個“假事務”,因為它並不支持回滾,也就是說在redis中一個事務有多個命令執行,並不能保證原子性。所以要使用redis的事務,一定要慎重

Redis中的“假事務”(不保證原子性)

在redis中事務相關的命令一共有以下幾個:

watch [key1] [key2]:監視一個或多個key,在事務開始之前如果被監視的key有改動,則事務被打斷。

multi:標記一個事務的開始。

exec:執行事務。

discard:取消事務的執行。

unwatch:取消監視的key。

  • 正常執行事務
127.0.0.1:6379> multi
OK
127.0.0.1:6379> set name kevin
QUEUED
127.0.0.1:6379> set age 25
QUEUED
127.0.0.1:6379> get name
QUEUED
127.0.0.1:6379> set sex male
QUEUED
127.0.0.1:6379> exec
1) OK
2) OK
3) "kevin"
4) OK
  • 取消事務執行

取消事務執行,命令將不會被執行。

127.0.0.1:6379> multi
OK
127.0.0.1:6379> set name yulinfeng
QUEUED
127.0.0.1:6379> set age 26
QUEUED
127.0.0.1:6379> discard
OK
127.0.0.1:6379> get name
"kevin"
  • 事務中的命令出現命令性錯誤,類似Java的編譯錯誤,執行事務時,所有的命令都不會被執行。
127.0.0.1:6379> multi
OK
127.0.0.1:6379> set name yulinfeng
QUEUED
127.0.0.1:6379> setget age 26
(error) ERR unknown command `setget`, with args beginning with: `age`, `26`, 
127.0.0.1:6379> exec
(error) EXECABORT Transaction discarded because of previous errors.
127.0.0.1:6379> get name
"kevin"
  • 事務中出現執行時錯誤,類似Java的運行時異常,執行事務時,部分命令會被執行成功,也即是不保證原子性
127.0.0.1:6379> multi
OK
127.0.0.1:6379> incr name
QUEUED
127.0.0.1:6379> set age 26
QUEUED
127.0.0.1:6379> exec
1) (error) ERR value is not an integer or out of range
2) OK
127.0.0.1:6379> get age
"26"
  • 使用watch監視key在事務之前被改動,正常未被改動時的情況,所有命令正常執行。
127.0.0.1:6379> watch name
OK
127.0.0.1:6379> multi
OK
127.0.0.1:6379> set name yulinfeng
QUEUED
127.0.0.1:6379> set age 18
QUEUED
127.0.0.1:6379> exec
1) OK
2) OK
127.0.0.1:6379> get age
"18"
  • 使用watch監視key,此時在事務執行前key被改動,事務將取消不會執行所有命令。

我們現在一個redis客戶端中執行watch命令。

127.0.0.1:6379> watch name
OK

此時我們打開另一個redis客戶端,修改key=name的值。

127.0.0.1:6379> set name kevin
OK

我們再次回到第一個客戶端,開始輸入事務的命令塊。

127.0.0.1:6379> multi
OK
127.0.0.1:6379> set name abc
QUEUED
127.0.0.1:6379> set age 1
QUEUED
127.0.0.1:6379> exec
(nil)

可看到通過exec執行事務時,事務並沒有執行成功,而是返回“nil”。

Java中Jedis使用redis事務,則通過調用以下方法實現,具體命令可參照文檔:

@Test
public void testTransaction() {
    Jedis jedis = RedisClient.getJedis();
    jedis.watch("a", "c");
    Transaction transaction = jedis.multi();
    transaction.set("a", "b");
    transaction.set("c", "d");
    transaction.exec();
}

通過Lua腳本保證Redis的真事務

redis中自帶的事務命令,最致命的前面已經多次提到,那就是不保證原子性,所以在使用redis的事務時,一定要謹慎。

但如果我們一定要在redis中實現真正的事務應該怎麼辦呢?redis為我們提供了另外一種更為“靈活”的方式——Lua腳本

在這裡當然並不會詳細講解Lua的語法規則,我們一步步來看在redis中如何執行Lua腳本,以及Lua是如何運用在redis保證事務的。

我們先用Lua腳本在redis中實現調用字元串的set命令,我們先看示例:

127.0.0.1:6379> eval "return redis.call('set', KEYS[1], ARGV[1])" 1 company bat
OK
127.0.0.1:6379> get company
"bat"

eval是執行Lua腳本的命令,第二個參數是Lua腳本,第三個參數是一個數字表示一共有多少個key,第四個參數表示key值,第五個參數表示value值,eval [lua scripts] [numskey] [key1] [key2] [value1] [value2] ……

接下來,我們來一個Lua腳本,腳本中包含寫入name的值和age的值。

127.0.0.1:6379> eval "redis.call('set', KEYS[1], ARGV[1]) redis.call('set', KEYS[2], ARGV[2])" 2 name age kevin 25
(nil)
127.0.0.1:6379> get name
"kevin"
127.0.0.1:6379> get age
"25"

對於簡單的Lua腳本通過命令行的方式直接編輯問題不大,但如果是比較複雜得Lua腳本,通常我們會單獨寫一個Lua腳本文件,然後載入它,例如以下示例:

local exist = redis.call('exists', KEYS[1])

if exist then
    return redis.call('incr', KEYS[1])
else
    return nil
end

我們將它保存為Lua腳本文件,執行以下命令:

okevindeMacBook-Air:redis-5.0.7 okevin$ redis-cli --eval ~/Desktop/lua_test.lua view
(nil)

可以看到key=view並不存在,所以返回nil,如果此時我們在redis中定義了一個key=view的值,此時將返回以下信息:

okevindeMacBook-Air:redis-5.0.7 okevin$ redis-cli --eval ~/Desktop/lua_test.lua view
(integer) 2

Jedis中如何載入Lua腳本

有關本節的源碼:https://github.com/yu-linfeng/redis5.x_tutorial/tree/master/code/jedis

在Jedis可以直接調用Jedis類的eval方法,第一個參數是Lua腳本,第二個參數是key值,第三個參數是value值。

public void testLua() {
    Jedis jedis = RedisClient.getJedis();
    List<String> keys = new ArrayList<>();
    keys.add("name");
    keys.add("age");
    List<String> values = new ArrayList<>();
    values.add("kevin");
    values.add("25");
    jedis.eval("redis.call('set', KEYS[1], ARGV[1]) redis.call('set', KEYS[2], ARGV[2])", keys, values);
    jedis.close();
}

第七章 · 分散式鎖

redis在我們日常開發中,除了用來做緩存提高應用程式的性能,降低資料庫壓力之外。可能用途最廣泛地當屬用redis來做分散式鎖了。

在單機中,我們要解決併發時線程安全的問題會使用JDK的synchronized或者Lock類,或者直接使用線程安全的類,例如JUC(java.util.concurrent併發包)。而在大型的應用程式中,單機部署顯然不能滿足我們的需求,這個時候要在分散式集群環境中對互斥資源進行控制訪問,就需要使用到分散式鎖。

在本章中,我們著重介紹基於redis的分散式鎖,同時將簡單介紹其他分散式鎖的解決方案。

開始之前先總結無論什麼方式的分散式鎖,其核心都是如有不存在某個key則寫入,存在則返回寫入失敗

通過redis實現分散式鎖

redis中主要通過setnx命令實現,全稱是“SET if Not eXists”,意為如果存在則寫入。如果不存在key則返回1,已經存在了這個key,則會返回0。釋放鎖時直接調用del命令刪除即可。

127.0.0.1:6379> setnx redis_lock a
(integer) 1
127.0.0.1:6379> setnx redis_lock a
(integer) 0

但是請註意,使用setnx有一定的風險,我們知道加鎖就有存在“死鎖”的可能性,而打破死鎖的方法之一就是主動釋放資源(設置鎖過期時間),然而setnx並沒有提供過期時間的設置,redis提供了另外一個命令——expire來設置key值得過期時間,所以改造上面的例子為以下所示:

127.0.0.1:6379> setnx redis_lock a          #設置一個分散式鎖的key為redis_lock
(integer) 1
127.0.0.1:6379> expire redis_lock 5         #設置redis_lock的過期時間為5秒,到期自動刪除
(integer) 1
127.0.0.1:6379> setnx redis_lock a          #此時再設置分散式鎖的key為redis_lock,返回0失敗
(integer) 0
127.0.0.1:6379> setnx redis_lock a          #過5秒再設置分散式鎖的key為redis_lock,返回1成功
(integer) 1

可以看到通過組合setnxexpire命令,能達到我們想要的結果。但是請註意,它仍然存在一個問題,那就是這兩個命令並不是原子性的,如果在執行expire redis_lock 5時,redis服務恰好宕機,此時這個key將會一直存在。

好在redis為我們提供了set命令的分散式用法並且可以設置為過期時間,關鍵是原子性的。官方的命令參數為set key value [expiration EX seconds|PX milliseconds] [NX|XX]

[expiration EX seconds|PX milliseconds]參數EX表示過期時間單位為“秒”,PX表示過期時間單位為“毫秒”。

[NX|XX]參數NX表示“SET if Not eXists”不存在則寫入,XX表示“SET if eXists”存在則寫入,分散式鎖的場景中使用“NX”參數。

所以我們設置一個key值名為“lock”的鎖,5秒後自動刪除:

127.0.0.1:6379> set lock a ex 5 nx          #設置一個key值名為“lock”的鎖,5秒後自動刪除
OK
127.0.0.1:6379> set lock a ex 5 nx          #5秒內設置一個key值名為“lock”的鎖,5秒後自動刪除。返回nil失敗
(nil)
127.0.0.1:6379> set lock a ex 5 nx          #5秒後設置一個key值名為“lock”的鎖,5秒後自動刪除。返OK成功
OK

使用redis作為分散式鎖,最好要設置過期時間,也就是最好使用set命令。

其他分散式鎖

通過ZooKeeper實現分散式鎖

ZooKeeper是一個分散式協調服務中間件,它可以用作註冊中心動態配置中心等等。

我們利用ZooKeeper的臨時有序節點也可以實現分散式鎖。

ZooKeeper的數據結構類似Linux中的文件結構,總體來講它時“一棵樹”,節點中記錄相關信息。節點分為“永久節點”和“臨時節點”。當我們要獲取一個鎖時,需要在ZooKeeper的結構中創建一個臨時有序節點,釋放鎖同樣時刪除節點。獲取分散式鎖,即獲取一個ZooKeeper的臨時有序節點,如果獲取到的有序節點存在比序號比自己更小的兄弟節點,即獲取鎖失敗。

基於ZooKeeper實現分散式鎖可以利用ZooKeeper監聽的特性,一旦有節點發生變化可以進行通知。這點是Redis不具備的。但由於它的實現方式是創建和刪除節點,所以在性能上不如redis。

通過MySQL實現分散式鎖

通過MySQL實現分散式鎖是我以前遇到的一個面試問題,思考以下實現方式:

在MySQL創建一個有關鎖的表“tb_lock”,一共有兩列,一列叫“key”並設置為唯一索引,另一列設置為“value”。

獲取鎖時,通過insert插入一條記錄,如果插入成功則獲取鎖成功;插入失敗則獲取鎖失敗。

一聽,是不是覺得有點意思,好像確實能通過MySQL來實現分散式鎖,這樣我們就不必引入redis或ZooKeeper。那為什麼我們日常開發中幾乎沒有人這樣用過呢?實際上,MySQL實現分散式鎖,它僅僅滿足了控制互斥資源這一點,儘管它是最核心的,但分散式鎖不僅是控制互斥資源,它還需要具備以下特性:

  • 可設置過期時間,防止死鎖
  • 需要具備阻塞獲取鎖的特性
  • 較高的性能和可靠性
  • 鎖還需要可重入
  • ……

所以如果要使用MySQL來實現分散式鎖,你需要去解決以上的問題,對於成熟的redis和ZooKeeper分散式鎖方案,我們大可不必再造一個不可靠的輪子。

關註公眾號:CoderBuff,回覆“redis”獲取《Redis5.x入門教程》完整版PDF。
這是一個能給程式員加buff的公眾號 (CoderBuff)


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 原創聲明 本文作者:黃小斜 轉載請務必在文章開頭註明出處和作者。 思維導圖 簡介 上一期我講到了數據結構怎麼學,當時我的學習方法是先看視頻,再看書,然後刷一刷基礎演算法題做鞏固。 作為一個非科班考研黨,當時覺得這個學習路線還是比較平滑的,畢竟能讓我一個小白快速學會,這門課相對來說還是比較簡單的。 於是 ...
  • 八、多態 多態(也稱作動態綁定、後期綁定或運行時綁定) 域(成員變數)是不具有多態性的,只有普通的方法調用是多態的,任何域訪問操作都將由編譯器解析,因此不是多態的 靜態方法也是不具有多態性的 public static void main(String[] args){ Super sup = ne ...
  • 在搭建個人博客時,大家都會買一臺雲伺服器。可是圖片的存放一直是一個問題,冷月幫大家找到一個免費的第三方平臺對象存儲 七牛雲。大家可以把圖片上傳到七牛雲的對象存儲,大大節約伺服器的壓力。 首先,大家在使用七牛雲的對象存儲必須到官網上申請一個賬號,並且實名認證。 "七牛雲官網" 進入個人中心,點擊秘鑰管 ...
  • 四、const限定符【引用/指針/頂層/常量表達式】 const對象值不變,必須初始化,能完成此type的大部分operation。 一般,多文件獨立變數,編譯初始化僅文件內有效; 除非,(條件:初值不是常量表達式 and 聲明+定義 都有 extern); //file.h extern cons ...
  • 一、JVM記憶體分配與回收 下圖為堆記憶體結構圖(註意:元數據區(MetaData )實際上不屬於堆): 1、對象優先在Eden區分配 大多數情況下,對象在新生代中Eden區分配。當Eden區沒有足夠空間進行分配時,JVM將發起一次Minor GC。 提問:Minor GC和Full GC有什麼不同呢? ...
  • 原創聲明 本文作者:黃小斜 轉載請務必在文章開頭註明出處和作者。 本文思維導圖 ​ 資料庫和關係型資料庫 作為一個程式員,不瞭解資料庫怎麼能行,那麼資料庫到底是個啥呢,作為一個Java工程師,平時和資料庫打交道著實不少,所謂的CRUD其實就是對資料庫進行增刪改查的操作。 根據百度百科的介紹,資料庫是 ...
  • 1.面向對象 面向對象的特點:封裝、繼承、多態、抽象 封裝:封裝是把過程和數據包圍起來,對數據的訪問只能通過已定義的介面。封裝是一種信息隱藏技術,在java中通過關鍵字private,protected和public實現封裝。 適當的封裝可以讓程式碼更容易理解和維護,也加強了程式碼的安全性。 繼承: ...
  • 1.abstract(抽象) 抽象含義:具有某種對象的特征,但不完整。(似是而非) 1.1 抽象類 語法: 在class關鍵字前面,加上abstract,代表這個類是一個抽象類 public abstract class Test{ ....... } 作用: 可被子類繼承,提供共性屬性和方法 可聲 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...