溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

php和redis實現加解鎖的方法

發布時間:2020-06-22 15:11:18 來源:億速云 閱讀:215 作者:Leah 欄目:關系型數據庫

這篇文章將為大家詳細講解有關php和redis實現加解鎖的方法,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

業務背景:在房間棋牌游戲中需要用到鎖來防止并發操作引起的 redis 數據臟讀問題;例如添加用戶進入房間的動作:

php和redis實現加解鎖的方法

并發的情況下,get RoomUsers 會有臟讀現象;

解決思路:加鎖房間來實現 一個房間每次只允許一個客戶端操作,其他并發客戶端則等待;也就是-----堵塞鎖;

加鎖:redis加鎖方式有幾種: incr、set、setnx、hSetnx,可以參考這篇文章:redis加鎖的幾種實現

這里我用到 set 這種方式

$roomId = $_GET['roomId'];
$user = $_GET['user'];             // '張三'
$key = "LockRoom:{$roomId}";
$value = $roomId.uniqid();
$ex = 3;
// 如果 $key 不存在的話,就設置 $key 的值為 $value,且有效期為 3s; 
// return TRUE / FALSE
while(true){
    $res  = $this->redis->set($key, $value, ['nx', 'ex' => $ex]);
    if($res) { break; }
    usleep(5000);
}

// 將用戶添加進房間
$roomUsers = $this->redis->get("Room:{$roomId}:Users"); // ['李四', '王五']
$roomUsers[] = $user;
$this->redis->set("Room:{$roomId}:Users", $roomUsers); // ['李四', '王五', '張三']

解鎖:操作完當然要解鎖了,不解鎖起碼要等待 3秒;

解鎖用 delete 刪除 key; 但是這里有個坑,不能直接用 delete,因為假設 client01 獲得了鎖,在添加用戶進入房間的過程中 時間超過了 3秒 ,這個時候client02 就會同樣獲得鎖并且設置3S,然后當client01 操作完之后 delete key , 就把 client02 設置的鎖刪除了;

這里推薦用 lua 代碼執行刪除,因為lua 執行具有原子性。

// 將用戶添加進房間
$roomUsers = $this->redis->get("Room:{$roomId}:Users"); // ['李四', '王五']
$roomUsers[] = $user;
$this->redis->set("Room:{$roomId}:Users", $roomUsers); // ['李四', '王五', '張三']

// lua 腳本解鎖
// 先判斷 key的值是否為 value, TRUE 才會刪除, 所以 $value 的設計要有隨機唯一性
$script = 'if redis.call("get",KEYS[1]) == ARGV[1]
then
    return redis.call("del",KEYS[1])
else
    return 0
end ';
$this->redis->eval($script, array($key , $value), 1);

關于php和redis實現加解鎖的方法就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女