Redis Watch 命令
作用:
用于監(jiān)視一個(gè)(或多個(gè)) key 屉栓,如果在事務(wù)執(zhí)行之前這個(gè)(或這些) key 被其他命令所改動(dòng)羊赵,那么事務(wù)將被打斷趟佃。
用法:
redis 127.0.0.1:6379> WATCH key1 key2
OK
Redis Unwatch 命令
作用:
用于取消 WATCH 命令對(duì)所有 key 的監(jiān)視。
用法:
redis 127.0.0.1:6379> UNWATCH
OK
Redis Multi 命令
作用:
用于標(biāo)記一個(gè)事務(wù)塊的開(kāi)始昧捷。事務(wù)塊內(nèi)的多條命令會(huì)按照先后順序被放進(jìn)一個(gè)隊(duì)列當(dāng)中闲昭,最后由 EXEC 命令原子性(atomic)地執(zhí)行。
用法:
redis 127.0.0.1:6379> MULTI # 標(biāo)記事務(wù)開(kāi)始
OK
redis 127.0.0.1:6379> INCR user_id # 多條命令按順序入隊(duì)
QUEUED
redis 127.0.0.1:6379> INCR user_id
QUEUED
redis 127.0.0.1:6379> INCR user_id
QUEUED
redis 127.0.0.1:6379> PING
QUEUED
redis 127.0.0.1:6379> EXEC # 執(zhí)行
1) (integer) 1
2) (integer) 2
3) (integer) 3
4) PONG
利用watch實(shí)現(xiàn)incr
具體做法如下:
WATCH mykey
val = GET mykey
val = val + 1
MULTI
SET mykey $val
EXEC
在獲取mykey的值之前先通過(guò)WATCH命令監(jiān)控了該鍵靡挥,此后又將set命令包圍在事務(wù)中序矩,這樣就可以有效的保證每個(gè)連接在執(zhí)行EXEC之前,如果當(dāng)前連接獲取的mykey的值被其它連接的客戶端修改跋破,那么當(dāng)前連接的EXEC命令將執(zhí)行失敗簸淀。這樣調(diào)用者在判斷返回值后就可以獲悉val是否被重新設(shè)置成功。
注意點(diǎn):
- 由于WATCH命令的作用只是當(dāng)被監(jiān)控的鍵值被修改后阻止之后一個(gè)事務(wù)的執(zhí)行毒返,而不能保證其他客戶端不修改這一鍵值租幕,所以在一般的情況下我們需要在EXEC執(zhí)行失敗后重新執(zhí)行整個(gè)函數(shù)。
- 執(zhí)行EXEC命令后會(huì)取消對(duì)所有鍵的監(jiān)控拧簸,如果不想執(zhí)行事務(wù)中的命令也可以使用UNWATCH命令來(lái)取消監(jiān)控劲绪。
示例:
打開(kāi)兩個(gè)redis-cli命令行窗口 session 1 和 session 2
session 1:
redis 127.0.0.1:6379> set test 1 # 設(shè)置test="1"
OK
redis 127.0.0.1:6379> get test # 獲取到test的值為"1"
"1"
redis 127.0.0.1:6379> watch test # 監(jiān)視test
OK
redis 127.0.0.1:6379> multi # 開(kāi)啟事務(wù)
OK
redis 127.0.0.1:6379> set test 2 # 將test設(shè)為"2"
QUEUED
redis 127.0.0.1:6379> exec # 待 session 2 執(zhí)行完畢后再執(zhí)行 session 1 的exec命令,發(fā)現(xiàn)執(zhí)行失敗
(nil)
redis 127.0.0.1:6379> get test # 獲取test的值,發(fā)現(xiàn)test值為 session 2 中所設(shè)置的"3"
"3"
redis 127.0.0.1:6379> unwatch # 取消監(jiān)視所有key
OK
redis 127.0.0.1:6379> set test 4 # 非事務(wù)變更test的值為"4"
OK
redis 127.0.0.1:6379> get test # 獲取到test="4"
"4"
session 2:
redis 127.0.0.1:6379> get test # 獲取到了 session 1 創(chuàng)建的test="1"
"1"
redis 127.0.0.1:6379> watch test # 監(jiān)視test
OK
redis 127.0.0.1:6379> multi # 開(kāi)啟事務(wù)
OK
redis 127.0.0.1:6379> set test 3 # 將test設(shè)為"3"
QUEUED
redis 127.0.0.1:6379> exec # 執(zhí)行事務(wù)
1) OK
redis 127.0.0.1:6379> get test # 獲取到test="3"
"3"