目錄
Redis 事務滿足 ACID?
首頁 資料庫 Redis 什麼是事務的ACID? Redis事務能實現ACID嗎?

什麼是事務的ACID? Redis事務能實現ACID嗎?

Jan 22, 2022 am 09:26 AM
acid redis redis事務

Redis 事務支援 ACID 嗎?以下這篇文章帶大家了解一下Redis的事務,介紹一下Redis實現事務的方法,聊聊Redis 事務支援 ACID嗎,希望對大家有幫助!

什麼是事務的ACID? Redis事務能實現ACID嗎?

騰訊面試官:「Redis 的事務了解麼?它的事務機制能實現ACID 屬性麼?」

程許遠:「撓頭,這個…我知道lua 腳本能實現事務…」

騰訊面試官:「好的,回去等通知吧。」


##碼哥,我斬獲了很多offer,沒想到最後敗在了「Redis 如何實現事務?」這個問題。

我們來一步一步分析:

  • 什麼是交易 ACID?

  • Redis 如何實作交易?

  • Redis 的事務能實現哪些屬性?

  • Lua 腳本實作。

什麼是事務的ACID

鬼吹燈之《雲南蟲谷》中的摸金校尉有句話叫「合則生,分則死」,為了尋找雮塵珠他們三人分工明確、

齊心協力共進退方可成功。

交易(Transaction)是並發控制單位,一個操作序列組合而成,這些操作要么都執行,要么都不執行。 【相關推薦:

Redis影片教學

「是一個不可分割的工作單位」。

事務在執行時,會提供專門的屬性保證:

  • 原子性(Atomicity):一個事務的多個操作必須完成,或者都不完成( ps:MySQL 的原子性靠什麼實現呢?歡迎留言區評論);

  • #一致性(Consistency):事務執行結束後,資料庫的完整性限制沒有被破壞,事務執行的前後順序都是合法資料狀態。

    資料庫的完整性限制包括但不限於:

      實體完整性(如行的主鍵存在且唯一);
    • 列完整性(如字段的類型、大小、長度要符合要求)
    • 外鍵約束;
    • 使用者自訂完整性(如轉帳前後,兩個帳戶餘額的和應該不變)。
  • 隔離性(Isolation):交易內部的操作與其他交易是隔離的,並發執行的各個事務之間不能互相干擾。

    講究的是不同事務之間的相互影響,嚴格的隔離性對應隔離等級中的可串行化(Serializable)。

  • 持久性(Durability):交易一旦提交,所有的修改將永久的保存到資料庫中,即使系統崩潰重啟後資料也不會遺失。

碼哥,了解了 ACID 的特定要求後,Redis 如何實現事務機制?

Redis 如何實作交易

MULTIEXECDISCARD WATCH 指令是Redis 實作交易的基礎。

Redis 交易的執行過程包含三個步驟:

  • 開啟交易;

  • 指令入隊;

  • 執行交易或丟棄;

#明確開啟一個交易

客戶端透過

MULTI 指令明確表示開啟一個事務,隨後的指令將排隊緩存,並不會實際執行。

命令入隊

客戶端把事務中的要執行的一系列指令傳送到服務端。

要注意的是,雖然指令傳送到服務端,但是 Redis 實例只是把這一系列指令暫存在一個指令佇列中,並不會立刻執行。

執行交易或丟棄

客戶端向服務端發送提交或丟棄交易的命令,讓Redis 執行第二步驟中發送的具體指令或清空佇列指令,放棄執行。

Redis 只要在呼叫

EXEC 時,即可安排佇列指令執行。

也可透過

DISCARD 丟棄第二步驟中保存在佇列中的指令。

Redis 事務案例

透過線上偵錯網站執行我們的範例程式碼:try.redis.io

#正常執行

透過MULTI EXEC 執行一個事務程序:

# 开启事务
> MULTI
OK
# 开始定义一些列指令
> SET “公众号:码哥字节” "粉丝 100 万"
QUEUED
> SET "order" "30"
QUEUED
> SET "文章数" 666
QUEUED
> GET "文章数"
QUEUED
# 实际执行事务
> EXEC
1) OK
2) OK
3) OK
4) "666"
登入後複製

我們看到每個讀寫指令執行後的回傳結果都是QUEUED,表示謝謝操作都被暫存到了命令隊列,還沒有實際執行。

當執行了 EXEC 指令,就可以看到具體每個指令的回應資料。

放棄交易

透過MULTIDISCARD丟棄佇列指令:

# 初始化订单数
> SET "order:mobile" 100
OK
# 开启事务
> MULTI
OK
# 订单 - 1
> DECR "order:mobile"
QUEUED
# 丢弃丢列命令
> DISCARD
OK
# 数据没有被修改
> GET "order:mobile"
"100"
登入後複製

#碼哥,Redis 的事務能保證ACID 特性麼?

這個問題問得好,我們一起來分析下。

Redis 事務滿足 ACID?

Redis 交易可以一次執行多個命令, 並且帶有以下三個重要的保證:

  • ##批次指令在執行EXEC 指令之前會放入佇列暫存;

  • 收到EXEC 指令後進入事務執行,事務中任意指令執行失敗,其餘的指令依然被執行;

  • 事務執行過程中,其他客戶端提交的命令不會插入到目前命令執行的序列中。

原子性

#碼哥,如果事務執行過程中發生錯誤了,原子效能保證麼?

在交易期間,可能遇到兩種指令錯誤:

    在執行
  • EXEC 指令前,發送的指令本身就錯誤。如下:
      參數數量錯誤;
    • 命令名稱錯誤,使用了不存在的命令;
    • 記憶體不足(Redis 實例使用
    • maxmemory指令配置內存限制)。
  • 在執行
  • EXEC 指令後,指令可能會失敗。例如,命令和操作的資料類型不符(對 String 類型 的 value 執行了 List 清單操作);
  • 在執行交易的
  • EXEC 命令時。 Redis 實例發生了故障導致交易執行失敗。

EXEC 執行前報錯誤

在命令入隊時,Redis 就會

報錯並且記錄下這個錯誤

此時,我們

還能繼續提交指令操作

等到執行了

EXEC指令之後,Redis 就會拒絕執行所有提交的命令操作,回傳事務失敗的結果

這樣一來,

事務中的所有指令都不會再被執行了,保證了原子性。

如下是指令入隊發生錯誤,導致交易失敗的例子:

#开启事务
> MULTI
OK
#发送事务中的第一个操作,但是Redis不支持该命令,返回报错信息
127.0.0.1:6379> PUT order 6
(error) ERR unknown command `PUT`, with args beginning with: `order`, `6`,
#发送事务中的第二个操作,这个操作是正确的命令,Redis把该命令入队
> DECR b:stock
QUEUED
#实际执行事务,但是之前命令有错误,所以Redis拒绝执行
> EXEC
(error) EXECABORT Transaction discarded because of previous errors.
登入後複製

EXEC 執行後報錯

事務操作入隊時,命令和操作的資料類型不匹配,但Redis 實例沒有檢查出錯誤。

但是,執行完 EXEC 指令以後,Redis 實際執行這些指令,就會報錯。

敲黑板了:Redis 雖然會對錯誤指令報錯,但是事務依然會把正確的命令執行完,這時候事務的原子性就無法保證了!

碼哥,為什麼 Redis 不支援回溯?

其實,Redis 中並沒有提供回溯機制。雖然 Redis 提供了 DISCARD 命令。

但是,這個指令只能用來主動放棄交易執行,把暫存的指令佇列清空,起不到回滾的效果。

EXEC 執行時,發生故障

如果 Redis 開啟了 AOF 日誌,那麼,只會有部分的交易操作被記錄到 AOF 日誌中。

我們需要使用 redis-check-aof 工具檢查 AOF 日誌文件,這個工具可以把未完成的事務操作從 AOF 文件中移除。

這樣一來,我們使用 AOF 恢復實例後,事務操作就不會再被執行,從而保證了原子性。

簡單總結

    命令入隊時就報錯,會放棄事務執行,保證原子性;
  • 命令入隊時沒報錯,實際執行時報錯,不保證原子性;
  • EXEC 指令執行時實例故障,如果開啟了AOF 日誌,可以保證原子性。

一致性

一致性會受到錯誤指令、執行個體故障發生時機的影響,依照指令出錯實例故障兩個維度的發生時機,可以分成三種情況來分析。

EXEC 執行前,入隊報錯

事務會被放棄執行,所以可以保證一致性。

EXEC 執行後,實際執行時報錯誤

有錯誤的執行不會執行,正確的指令可以正常執行,一致性可以保證。

EXEC 執行時,實例故障

實例故障後會進行重啟,這就和資料復原的方式有關了,我們要根據實例是否開啟了RDB 或AOF 來分情況討論下。

如果我們沒有開啟 RDB 或 AOF,那麼,在實例故障重新啟動後,資料都沒有了,資料庫是一致的。

如果我們使用了 RDB 快照,因為 RDB 快照不會在交易執行時執行。

所以,交易指令操作的結果不會被儲存到 RDB 快照中,使用 RDB 快照進行復原時,資料庫裡的資料也是一致的。

如果我們使用了 AOF 日誌,而交易操作還沒有被記錄到 AOF 日誌時,實例就發生了故障,那麼,使用 AOF 日誌復原的資料庫資料是一致的。

如果只有部分操作被記錄到了 AOF 日誌,我們可以使用 redis-check-aof 清除交易中已經完成的操作,資料庫復原後也是一致的。

隔離性

交易執行又可以分成指令入隊(EXEC 指令執行前)和指令實際執行(EXEC 指令執行後)兩個階段。

所以在並發執行的時候我們針對這兩個階段分兩種情況分析:

  • 並發操作在EXEC 指令前執行,隔離性需要透過WATCH 機制保證;

  • 並發操作在EXEC 指令之後,隔離性可以保證。

碼哥,什麼是 WATCH 機制?

我們重點來看第一種情況:當一個事務的 EXEC 指令還沒執行時,事務的指令操作是暫存在指令佇列中的。

此時,如果有其它的並發操作,同樣的 key 被修改,需要看交易是否使用了 WATCH 機制。

WATCH 機制的作用是:在交易執行前,監控一個或多個鍵的值變化情況,當交易呼叫EXEC 指令執行時,WATCH 機制會先檢查監控的鍵是否被其它客戶端修改了。

如果修改了,就放棄交易執行,避免事務的隔離性被破壞。

同時,客戶端可以再次執行事務,此時,如果沒有並發修改事務資料的操作了,事務就能正常執行,隔離性也得到了保證。

什麼是事務的ACID? Redis事務能實現ACID嗎?

沒有 WATCH

如果沒有 WATCH 機制, 在 EXEC 指令執行前的並發操作對資料讀寫。

當執行 EXEC 的時候,事務內部要操作的資料已經改變,Redis 並沒有做到事務之間的隔離。

什麼是事務的ACID? Redis事務能實現ACID嗎?

並發操作在EXEC 之後接收執行

至於第二種情況,因為Redis 是用單執行緒執行指令,而且,EXEC 指令執行後,Redis 會保證先把指令佇列中的所有指令執行完再執行之後的指令。

所以,在這種情況下,並發操作不會破壞交易的隔離性。

什麼是事務的ACID? Redis事務能實現ACID嗎?

持久性

#如果Redis 沒有使用RDB 或AOF,那麼交易的持久化屬性肯定得不到保證。

如果Redis 使用了RDB 模式,那麼,在一個事務執行後,而下一次的RDB 快照還未執行前,如果發生了實例宕機,資料遺失,這種情況下,事務修改的數據也是不能保證持久化的。

如果 Redis 採用了 AOF 模式,因為 AOF 模式的三個設定選項 no、everysec 和 always 都會存在資料遺失的情況。

所以,事務的持久性屬性也還是得不到保證。

不管 Redis 採用什麼持久化模式,交易的持久性屬性是得不到保證的。

總結

  • #Redis 具備了一定的原子性,但不支援回滾。
  • Redis 不具備 ACID 中一致性的概念。 (或者說 Redis 在設計時就無視這點)
  • Redis 具備隔離性。
  • Redis 無法保證持久性。

Redis 的交易機制可以保證一致性和隔離性,但無法保證持久性。

不過,因為 Redis 本身就是記憶體資料庫,持久性並不是一個必須的屬性,我們更加關注的還是原子性、一致性和隔離性這三個屬性。

原子性的情況比較複雜,當事務中使用的命令語法有誤時,原子性得不到保證,在其它情況下,事務都可以原子性執行。

更多程式相關知識,請造訪:程式設計入門! !

以上是什麼是事務的ACID? Redis事務能實現ACID嗎?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

redis集群模式怎麼搭建 redis集群模式怎麼搭建 Apr 10, 2025 pm 10:15 PM

Redis集群模式通過分片將Redis實例部署到多個服務器,提高可擴展性和可用性。搭建步驟如下:創建奇數個Redis實例,端口不同;創建3個sentinel實例,監控Redis實例並進行故障轉移;配置sentinel配置文件,添加監控Redis實例信息和故障轉移設置;配置Redis實例配置文件,啟用集群模式並指定集群信息文件路徑;創建nodes.conf文件,包含各Redis實例的信息;啟動集群,執行create命令創建集群並指定副本數量;登錄集群執行CLUSTER INFO命令驗證集群狀態;使

redis數據怎麼清空 redis數據怎麼清空 Apr 10, 2025 pm 10:06 PM

如何清空 Redis 數據:使用 FLUSHALL 命令清除所有鍵值。使用 FLUSHDB 命令清除當前選定數據庫的鍵值。使用 SELECT 切換數據庫,再使用 FLUSHDB 清除多個數據庫。使用 DEL 命令刪除特定鍵。使用 redis-cli 工具清空數據。

redis怎麼讀取隊列 redis怎麼讀取隊列 Apr 10, 2025 pm 10:12 PM

要從 Redis 讀取隊列,需要獲取隊列名稱、使用 LPOP 命令讀取元素,並處理空隊列。具體步驟如下:獲取隊列名稱:以 "queue:" 前綴命名,如 "queue:my-queue"。使用 LPOP 命令:從隊列頭部彈出元素並返回其值,如 LPOP queue:my-queue。處理空隊列:如果隊列為空,LPOP 返回 nil,可先檢查隊列是否存在再讀取元素。

centos redis如何配置Lua腳本執行時間 centos redis如何配置Lua腳本執行時間 Apr 14, 2025 pm 02:12 PM

在CentOS系統上,您可以通過修改Redis配置文件或使用Redis命令來限制Lua腳本的執行時間,從而防止惡意腳本佔用過多資源。方法一:修改Redis配置文件定位Redis配置文件:Redis配置文件通常位於/etc/redis/redis.conf。編輯配置文件:使用文本編輯器(例如vi或nano)打開配置文件:sudovi/etc/redis/redis.conf設置Lua腳本執行時間限制:在配置文件中添加或修改以下行,設置Lua腳本的最大執行時間(單位:毫秒)

redis指令怎麼用 redis指令怎麼用 Apr 10, 2025 pm 08:45 PM

使用 Redis 指令需要以下步驟:打開 Redis 客戶端。輸入指令(動詞 鍵 值)。提供所需參數(因指令而異)。按 Enter 執行指令。 Redis 返迴響應,指示操作結果(通常為 OK 或 -ERR)。

redis命令行怎麼用 redis命令行怎麼用 Apr 10, 2025 pm 10:18 PM

使用 Redis 命令行工具 (redis-cli) 可通過以下步驟管理和操作 Redis:連接到服務器,指定地址和端口。使用命令名稱和參數向服務器發送命令。使用 HELP 命令查看特定命令的幫助信息。使用 QUIT 命令退出命令行工具。

redis過期策略怎麼設置 redis過期策略怎麼設置 Apr 10, 2025 pm 10:03 PM

Redis數據過期策略有兩種:定期刪除:定期掃描刪除過期鍵,可通過 expired-time-cap-remove-count、expired-time-cap-remove-delay 參數設置。惰性刪除:僅在讀取或寫入鍵時檢查刪除過期鍵,可通過 lazyfree-lazy-eviction、lazyfree-lazy-expire、lazyfree-lazy-user-del 參數設置。

如何優化debian readdir的性能 如何優化debian readdir的性能 Apr 13, 2025 am 08:48 AM

在Debian系統中,readdir系統調用用於讀取目錄內容。如果其性能表現不佳,可嘗試以下優化策略:精簡目錄文件數量:盡可能將大型目錄拆分成多個小型目錄,降低每次readdir調用處理的項目數量。啟用目錄內容緩存:構建緩存機制,定期或在目錄內容變更時更新緩存,減少對readdir的頻繁調用。內存緩存(如Memcached或Redis)或本地緩存(如文件或數據庫)均可考慮。採用高效數據結構:如果自行實現目錄遍歷,選擇更高效的數據結構(例如哈希表而非線性搜索)存儲和訪問目錄信

See all articles