高級Java程式員必問,Redis事務終極篇

来源:https://www.cnblogs.com/yidengjiagou/archive/2023/05/05/17373712.html
-Advertisement-
Play Games

Redis事務(Transaction)通過將多個Redis操作封裝為一個原子性的操作序列,確保在事務執行過程中,不會受到其他客戶端的干擾。從而在保證數據一致性的同時,協調併發,提高數據操作的效率和性能 ...


1. 簡介

1.1 什麼是Redis事務

Redis事務(Transaction)通過將多個Redis操作封裝為一個原子性的操作序列,確保在事務執行過程中,不會受到其他客戶端的干擾。從而在保證數據一致性的同時,協調併發,提高數據操作的效率和性能。

1.2 Redis事務的應用場景

在分散式系統和高併發場景下,事務處理具有重要意義。Redis事務可以確保數據的一致性,避免併發操作導致的數據不一致問題。以下是一些Redis事務的應用場景:

  1. 批量操作:Redis 事務可以將多個命令打包成一個單元來執行,可以減少與 Redis 伺服器的通信次數,從而提高性能。
  2. 資料庫遷移:在遷移數據時,需要保證數據一致性。通過Redis事務,可以確保數據在遷移過程中不會出現不一致的情況。
  3. 分散式鎖:在分散式系統中,為了保證數據的一致性,需要實現分散式鎖。通過Redis事務,可以在同一個事務中執行鎖定、解鎖等操作,確保鎖的原子性。

這些應用場景展示了Redis事務在實際應用中的價值。接下來,我們將詳細介紹Redis事務的基本命令、特性和實現原理。

2. Redis事務基本命令

在Redis中,事務的處理主要涉及以下五個基本命令:

2.1 MULTI

MULTI 命令用於標記一個事務塊的開始。在執行 MULTI 之後,Redis將開始記錄後續的命令,並將這些命令放入一個隊列中,直到遇到 EXEC 命令。

2.2 EXEC

EXEC 命令用於觸發事務塊中的所有命令一起執行。當Redis收到 EXEC 命令後,它將按照FIFO(先進先出)的順序執行事務隊列中的所有命令。如果事務執行成功,Redis會返回一個數組,其中包含每個命令執行後的結果。如果事務執行失敗,Redis將返回一個錯誤信息。

2.3 DISCARD

DISCARD 命令用於取消一個事務塊。當執行 DISCARD 命令後,Redis將清空事務隊列,並恢復到正常執行模式。任何在事務塊中的命令都不會被執行。

2.4 WATCH

WATCH 命令用於監視一個或多個Key,以確保在事務執行期間,這些Key的值沒有發生變化。如果在事務執行之前,有其他客戶端修改了這些被監視的Key,那麼事務將被中斷,並返回一個錯誤。這種機制被稱為樂觀鎖(Optimistic Locking)。

2.5 UNWATCH

UNWATCH 命令用於取消對所有Key的監視。執行 UNWATCH 後,Redis將不再監視任何Key的變化,事務將按照正常流程執行。

通過這五個基本命令,Redis實現了事務功能。接下來,我們將詳細介紹Redis事務的特性、實現原理以及在實際應用中的案例。

3. Redis事務的使用

下麵演示一個常見的電商購物場景,把更新訂單狀態和扣庫存放在一個事務中。

# 開啟事務
> MULTI
OK

# 執行命令
# 1. 設置訂單狀態為已完成
> SET order_status 1
QUEUED
# 2. 庫存減一
> DECR stock
QUEUED
# 3. 查看庫存
> GET stock
QUEUED

# 提交事務
> EXEC
1) OK
2) OK
3) 99

4. Redis事務的實現原理

4.1 事務隊列

當客戶端發送 MULTI 命令後,Redis開始記錄後續的命令,並將這些命令放入一個隊列中。當遇到 EXEC 命令時,Redis會按照FIFO(先進先出)的順序執行隊列中的所有命令。

4.2 錯誤處理

在事務執行過程中,可能會遇到命令執行失敗的情況。對於錯誤的處理,Redis採用的策略是:即使某個命令執行失敗,事務中的其他命令仍然會繼續執行。然而,整個事務的返回結果會包含錯誤信息,以便客戶端瞭解事務執行過程中發生的錯誤。

4.3 WATCH命令與樂觀鎖

WATCH 命令允許客戶端監視一個或多個Key,以確保在事務執行期間,這些Key的值沒有發生變化。這種機制被稱為樂觀鎖(Optimistic Locking)。如果在事務執行之前,有其他客戶端修改了這些被監視的Key,那麼事務將被中斷,並返回一個錯誤。樂觀鎖可以在一定程度上解決併發場景下的數據一致性問題。

5. Redis事務的註意事項與局限性

雖然Redis事務具有一定的功能,但在使用過程中需要註意以下事項:

5.1 無回滾機制

與傳統關係型資料庫不同,Redis事務不支持回滾(Rollback)。當事務中的某個命令執行失敗時,Redis不會回滾已執行的命令。因此,在使用Redis事務時,需要確保事務中的每個命令都能正確執行,以避免數據不一致的問題。

5.2 事務內的命令不支持條件判斷

Redis事務不支持在事務內進行條件判斷。這意味著,事務中的所有命令都會被執行,無論前面的命令是否執行成功。這可能導致數據的不一致性。想要解決這個問題,可以使用Lua腳本來實現條件判斷。

5.3 性能影響

由於Redis使用單線程模型來執行事務,因此,在事務執行期間,伺服器無法處理其他客戶端的請求。這可能對Redis的性能產生影響。為了降低事務對性能的影響,建議將事務中的命令數量控制在一個合理的範圍內。

5.4 ACID特性

Redis事務並不能完全保證事務四大特性,使用的時候需要註意:

  • 原子性:Redis事務具有一定的原子性,但是不支持回滾。
  • 一致性:Redis事務保證一致性。
  • 隔離性:Redis事務保證隔離性。Redis是單線程,事務執行期間,禁止其他客戶端發送命令給 Redis伺服器。
  • 持久性:Redis事務不保證持久性。Redis持久化機制都是非同步刷盤,存在數據丟失的情況。

6. 使用Lua腳本優化Redis事務

在某些場景下,Redis事務可能無法滿足應用的需求,例如需要在事務中進行條件判斷或迴圈。在這種情況下,可以使用Redis的Lua腳本功能來優化事務。Lua腳本可以在Redis伺服器端原子性地執行一系列命令,並支持條件判斷和迴圈,從而提供更強大的事務處理能力。

6.1 Lua腳本的基本使用

要在Redis中使用Lua腳本,可以使用EVAL命令執行腳本。例如,以下Lua腳本用於實現原子性地遞增一個計數器:

EVAL "local current = redis.call('get', KEYS[1]); current = current + 1; redis.call('set', KEYS[1], current); return current;" counter

6.2 Lua腳本與Redis事務的比較

與Redis事務相比,Lua腳本具有以下優勢:

  1. 更強大的邏輯處理能力:Lua腳本支持條件判斷、迴圈等複雜邏輯,而Redis事務只能順序執行命令。
  2. 更好的性能:由於Lua腳本在伺服器端執行,避免了多次往返通信帶來的延遲,因此性能通常優於Redis事務。
  3. 更高的可維護性:將業務邏輯封裝在Lua腳本中,可以提高代碼的可讀性和可維護性。

然而,使用Lua腳本也有一些局限性:

  1. 學習成本:使用Lua腳本需要學習Lua語言及其在Redis中的使用方法。
  2. 腳本管理:當業務邏輯變得複雜時,需要對多個Lua腳本進行維護和管理。
  3. 腳本執行的限制:為了避免長時間執行的腳本阻塞Redis伺服器,Redis對Lua腳本執行時間有一定的限制。如果腳本執行時間過長,可能會被強制終止。

7. 總結

本文主要介紹了Redis事務的概念、應用場景、基本命令、實現原理以及在實際應用中的案例。需要註意的是Redis事務並沒有完全實現事務的ACID特性,無回滾機制、也不支持條件判斷,可以使用Lua腳本優化Redis事務。

我是「一燈架構」,如果本文對你有幫助,歡迎各位小伙伴點贊、評論和關註,感謝各位老鐵,我們下期見

image


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

-Advertisement-
Play Games
更多相關文章
  • ChatGPT很強大,可以幫我們處理很多問題,但這些問題的答案的正確性您是否有考證過呢? 昨晚,DD就收到了一個有趣的反饋: 提問:有什麼關於數據許可權設計的資料推薦嗎? ChatGPT居然介紹了一本根本不存在的書《數據許可權設計與實現》,作者居然還是我... 那麼你在使用ChatGPT的時候,有碰到過 ...
  • 本文介紹將Windows電腦中的Administrator、網路、回收站等系統自帶桌面圖標取消顯示或恢復顯示的方法。 在Windows10電腦中,一般在桌面上預設會顯示如下所示的一些系統自帶圖標。 然而,在上述這些圖標中,有一些我們可能相對而言使用的頻率比較低,比如網路圖標,以及上圖中最上面的Adm ...
  • 本文源碼部分基於內核 5.4 版本討論 在經過上篇文章 《從內核源碼看 slab 記憶體池的創建初始化流程》 的介紹之後,我們最終得到下麵這幅 slab cache 的完整架構圖: 本文筆者將帶大家繼續從內核源碼的角度繼續拆解 slab cache 的實現細節,接下來筆者會基於上面這幅 slab ca ...
  • 哈嘍大家好我是鹹魚,在《Linux 記憶體管理 pt.1》中我們學習了什麼是物理記憶體、虛擬記憶體,瞭解了記憶體映射、缺頁異常等內容 那麼今天我們來接著學習 Linux 記憶體管理中的多級頁表和大頁 多級頁表&大頁 在《Linux 記憶體管理 pt.1》中我們知道了內核為每個進程都維護了一張頁表,這張頁表用來記 ...
  • 在MySQL中,這幾個都是統計操作,很多人在使用的時候,都使用的是count(1),這有沒有問題?使用正確?達到了統計效果? 我們從效果和效率兩方面來分析下 執行效果 count(*) 包括了所有的列,在統計時不會忽略列值為null的數據count(1) 用1表示代碼行,在統計時不會忽略列值為nul ...
  • 4月20日,袋鼠雲成功舉行了以“數實融合,韌性生長”為主題的2023春季生長大會。會上重磅發佈了袋鼠雲生態伙伴計劃——“飛躍計劃2.0”,從商機、產品、聯合方案及數據業務服務層面,與合作伙伴強強聯手,共同打造數字化生態,同時在聯合營銷、渠道政策、賦能培訓、產品開放、技術服務、交付實施等方面全面升級夥 ...
  • 摘要:本文主要介紹GaussDB(DWS)網路流控能力,並對其管控效果進行驗證。 本文分享自華為雲社區《GaussDB(DWS)網路流控與管控效果》,作者:門前一棵葡萄樹。 上一篇博文GaussDB(DWS)網路調度與隔離管控能力,我們詳細介紹了GaussDB網路調度邏輯,並簡單介紹瞭如何應用網路隔 ...
  • 0. 前情提要 系統的某個用來上報數據的介面存在死鎖的問題。這個介面內部對多張表進行了Update操作,執行順序為A表、B表、C表、D表、A表。死鎖發生的SQL,一條是第一次更新A表的SQL,另一條是第二次更新A表的SQL。整個更新都處在一個事務內,理論上講,只要第一個Session開始執行事務,第 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...