摘要:將用戶添加進(jìn)房間李四王五李四王五張三腳本解鎖先判斷的值是否為才會(huì)刪除所以的設(shè)計(jì)要有隨機(jī)唯一性具體還可以看看這篇文章解鎖鎖的正確姿勢(shì)還有操作的文檔里面有函數(shù)的解釋注意用腳本這里需要開(kāi)放等系統(tǒng)函數(shù)以上代碼僅作參考
php+redis 實(shí)現(xiàn)加鎖與解鎖操作
業(yè)務(wù)背景:在房間棋牌游戲中需要用到鎖來(lái)防止并發(fā)操作引起的 redis 數(shù)據(jù)臟讀問(wèn)題;例如添加用戶進(jìn)入房間的動(dòng)作:
并發(fā)的情況下,get RoomUsers 會(huì)有臟讀現(xiàn)象;
解決思路:加鎖房間來(lái)實(shí)現(xiàn) 一個(gè)房間每次只允許一個(gè)客戶端操作,其他并發(fā)客戶端則等待;也就是-----堵塞鎖;
加鎖:redis加鎖方式有幾種: incr、set、setnx、hSetnx,可以參考這篇文章:redis加鎖的幾種實(shí)現(xiàn)
這里我用到 set 這種方式
$roomId = $_GET["roomId"]; $user = $_GET["user"]; // "張三" $key = "LockRoom:{$roomId}"; $value = $roomId.uniqid(); $ex = 3; // 如果 $key 不存在的話,就設(shè)置 $key 的值為 $value,且有效期為 3s; // return TRUE / FALSE while(true){ $res = $this->redis->set($key, $value, ["nx", "ex" => $ex]); if($res) { break; } usleep(5000); } // 將用戶添加進(jìn)房間 $roomUsers = $this->redis->get("Room:{$roomId}:Users"); // ["李四", "王五"] $roomUsers[] = $user; $this->redis->set("Room:{$roomId}:Users", $roomUsers); // ["李四", "王五", "張三"]
解鎖:操作完當(dāng)然要解鎖了,不解鎖起碼要等待 3秒;
解鎖用 delete 刪除 key; 但是這里有個(gè)坑,不能直接用 delete,因?yàn)榧僭O(shè) client01 獲得了鎖,在添加用戶進(jìn)入房間的過(guò)程中 時(shí)間超過(guò)了 3秒 ,這個(gè)時(shí)候client02 就會(huì)同樣獲得鎖并且設(shè)置3S,然后當(dāng)client01 操作完之后 delete key , 就把 client02 設(shè)置的鎖刪除了;
這里推薦用 lua 代碼執(zhí)行刪除,因?yàn)閘ua 執(zhí)行具有原子性。
// 將用戶添加進(jìn)房間 $roomUsers = $this->redis->get("Room:{$roomId}:Users"); // ["李四", "王五"] $roomUsers[] = $user; $this->redis->set("Room:{$roomId}:Users", $roomUsers); // ["李四", "王五", "張三"] // lua 腳本解鎖 // 先判斷 key的值是否為 value, TRUE 才會(huì)刪除, 所以 $value 的設(shè)計(jì)要有隨機(jī)唯一性 $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);
具體還可以看看 這篇文章:解鎖 Redis 鎖的正確姿勢(shì)
還有php操作redis的文檔:PhpRedis 里面有 set()、eval() 函數(shù)的解釋
注意:用 lua 腳本這里 php.ini 需要開(kāi)放 shell_exec() 等系統(tǒng)函數(shù)
以上代碼僅作參考??!
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/28856.html
摘要:本篇博客將介紹第二種方式,基于的實(shí)現(xiàn)分布式鎖??偨Y(jié)本文主要介紹了如何使用代碼正確實(shí)現(xiàn)分布式鎖,對(duì)于加鎖和解鎖也分別給出了兩個(gè)比較經(jīng)典的錯(cuò)誤示例。其實(shí)想要通過(guò)實(shí)現(xiàn)分布式鎖并不難,只要保證能滿足可靠性里的四個(gè)條件。 前言 分布式鎖一般有三種實(shí)現(xiàn)方式:1.數(shù)據(jù)庫(kù)樂(lè)觀鎖;2、基于Redis的分布式鎖;3.基于Zookeeper的分布式鎖。本篇博客將介紹第二種方式,基于Redis的實(shí)現(xiàn)分布式鎖。...
摘要:貓說(shuō)多線程之內(nèi)存可見(jiàn)性下篇?dú)g迎你留言討論屬于你的見(jiàn)解,畢竟每個(gè)人的味蕾都不一樣,這杯咖啡有吸引到你嗎好像又是一個(gè)槽糕的比喻本文已轉(zhuǎn)載個(gè)人技術(shù)公眾號(hào)歡迎留言討論與點(diǎn)贊上一篇推薦貓說(shuō)主數(shù)據(jù)類型和引用下一篇推薦貓說(shuō)多線程之內(nèi)存可見(jiàn)性下篇 閱讀本文約3分鐘 本文大致講述兩種線程實(shí)現(xiàn)的可見(jiàn)性,或許你已經(jīng)提前想到了,那說(shuō)明你的基礎(chǔ)很好,我們要聊聊synchronized實(shí)現(xiàn)可見(jiàn)性與volatil...
閱讀 1217·2021-09-30 09:47
閱讀 3769·2021-09-06 15:02
閱讀 1781·2021-09-01 10:46
閱讀 2365·2019-08-30 15:52
閱讀 601·2019-08-29 15:28
閱讀 1875·2019-08-29 15:08
閱讀 1155·2019-08-29 13:28
閱讀 2580·2019-08-29 12:19