redis事務不具有回滾機制,那麼它是如何進行事務控制的

redis事務

任何數據庫都要有一套自己的事務控制機制,redis事務是一次可以執行多個命令,它的本質是一組命令的集合。一個事務中所有的命令都會被序列化,在事務執行的過程中會按照順序執行隊列中的命令。其它客戶端提交的命令請求會等到事務執行完畢再執行。

總的來說:redis事務就是一次性、順序性、排他性的執行一個隊列中的一系列命令。

redis事務和其它數據庫事務的區別:

1、redis事務是分為三個階段:開始事務、命令入隊、執行事務。

2、redis事務不具有隔離級別的概念:redis在發送exec命令之前,命令操作只是被放入到隊列緩存當中,並不會被實際執行,因此也就不能類似關係型數據中,在事務內查詢已經變更的操作,事務外的客戶端更不能查詢到事務內的數據。

3、redis事務是不保證原子性的:redis事務只保證在命令格式只有在都正確的情況下才會都執行,要不就都不執行命令。但是事務的整體是不保證原子性的,且沒有回滾,當事務中任意一個命令執行失敗,其餘的命令依然會執行。

redis命令語法結構:

1、watch key1 key2等等:監視一個或者多個KEY,如果在事務執行的時候,key的值被其它命令改動,則事務被打斷,全部不執行,redis通過該機制完成事務的樂觀鎖。

2、multi:用於指定redis事務的開始。

3、exec:用於指定redis事務開始執行(順序、一次性執行所有事務中的命令),一旦執行exec,前面加的監控鎖都會被取消。

4、discard:用於取消事務。放棄事務中的所有命令。

5、unwatch:取消對watch中key的監控。

下面我們通過實例來詳細分析redis事務的執行過程:

例1、redis正常事務流程:

multi
set key1 hello
set key2 free
set key3 world
get key2
exec
redis事務不具有回滾機制,那麼它是如何進行事務控制的

redis正常流程

例2、取消事務,代碼如下:

multi
set key1 hello_1
set key2 free_1
set key3 world_1
discard
get key3
redis事務不具有回滾機制,那麼它是如何進行事務控制的

redis取消事務

例3、redis事務中某個命令出錯(即不存在的命令,不是語法出錯)時,事務中的所有命令都不會執行,代碼如下:

multi
set key1 hello_1
setok key2 free_1
set key3 world_1
get key3
exec
redis事務不具有回滾機制,那麼它是如何進行事務控制的

redis事務

​例4、redis事務中存在某個命令具有語法性錯誤,執行exec時,其它命令照樣執行,代碼如下:

multi
incr key1 
set key2 free_1
set key3 world_1
get key3
exec
redis事務不具有回滾機制,那麼它是如何進行事務控制的

redis事務

例5、利用watch監控某個key值的變化,來做redis事務的樂觀鎖。模擬keya賬戶中有100塊錢,keyb賬戶有20塊錢,然後a向b轉了50塊錢。

首先添加測試數據,代碼如下:

set keya 100
set keyb 20 
watch keya 
multi
decrby keya 50
incrby keyb 50
exec 
get keya 
redis事務不具有回滾機制,那麼它是如何進行事務控制的

redis watch

然後,我們打開第二個客戶端,在事務執行(exec)之前,對a賬戶提前減少50塊錢,看下watch的樂觀鎖機制是否生效,代碼如下:

redis事務不具有回滾機制,那麼它是如何進行事務控制的

redis watch

在redis事務提交時,如果在事務進入緩存隊列的過程中,watch命令監控key的值發生了變化,則事務中的命令將都不會被執行。同時返回提示事務使用者事務執行失敗。


分享到:


相關文章: