解鎖redis鎖的正確姿勢
解鎖redis鎖的正確姿勢
redis是php的好朋友,在php寫業務過程中,有時候會使用到鎖的概念,同時只能有一個人可以操作某個行為。這個時候我們就要用到鎖。鎖的方式有好幾種,php不能在記憶體中用鎖,不能使用zookeeper加鎖,使用資料庫做鎖又消耗比較大,這個時候我們一般會選用redis做鎖機制。
setnx
鎖在redis中最簡單的資料結構就是string。最早的時候,上鎖的操作一般使用setnx,這個命令是當:lock不存在的時候set一個val,或許你還會記得使用expire來增加鎖的過期,解鎖操作就是使用del命令,虛擬碼如下:
if (Redis::setnx("my:lock", 1)) {
Redis::expire("my:lock", 10);
// ... do something
Redis::del("my:lock")
}
這裡其實是有問題的,問題就在於setnx和expire中間如果遇到crash等行為,可能這個lock就不會被釋放了。於是進一步的優化方案可能是在lock中儲存timestamp。判斷timestamp的長短。
set
現在官方建議直接使用set來實現鎖。我們可以使用set命令來替代setnx,就是下面這個樣子
if (Redis::set("my:lock", 1, "nx", "ex", 10)) {
... do something
Redis::del("my:lock")
}
上面的程式碼把my:lock設定為1,當且僅當這個lock不存在的時候,設定完成之後設定過期時間為10。
獲取鎖的機制是對了,但是刪除鎖的機制直接使用del是不對的。因為有可能導致誤刪別人的鎖的情況。
比如,這個鎖我上了10s,但是我處理的時間比10s更長,到了10s,這個鎖自動過期了,被別人取走了,並且對它重新上鎖了。那麼這個時候,我再呼叫Redis::del就是刪除別人建立的鎖了。
官方對解鎖的命令也有建議,建議使用lua指令碼,先進行get,再進行del
程式變成:
$token = rand(1, 100000);
function lock() {
return Redis::set("my:lock", $token, "nx", "ex", 10);
}
function unlock() {
$script = `
if redis.call("get",KEYS[1]) == ARGV[1]
then
return redis.call("del",KEYS[1])
else
return 0
end
`
return Redis::eval($script, "my:lock", $token)
}
if (lock()) {
// do something
unlock();
}
這裡的token是一個隨機數,當lock的時候,往redis的my:lock中存的是這個token,unlock的時候,先get一下lock中的token,如果和我要刪除的token是一致的,說明這個鎖是之前我set的,否則的話,說明這個鎖已經過期,是別人set的,我就不應該對它進行任何操作。
所以:不要再使用setnx,直接使用set進行鎖實現。
本文轉自軒脈刃部落格園部落格,原文連結:http://www.cnblogs.com/yjf512/p/6597814.html,如需轉載請自行聯絡原作者
相關文章
- 解鎖 Redis 鎖的正確姿勢Redis
- 掌握Redis分散式鎖的正確姿勢Redis分散式
- 用Python解鎖“吃雞”正確姿勢Python
- redis應用系列一:分散式鎖正確實現姿勢Redis分散式
- Redis的正確使用姿勢Redis
- 這才是實現分散式鎖的正確姿勢!分散式
- 「分散式」實現分散式鎖的正確姿勢?!分散式
- 【新夢想老師分享】分散式鎖的正確"姿勢"分散式
- 解鎖跨域的九種姿勢跨域
- Redis 分散式鎖的正確開啟方式Redis分散式
- Redis分散式鎖的正確實現方式Redis分散式
- 解鎖Spring框架姿勢1Spring框架
- Postman 正確使用姿勢Postman
- git commit 的正確姿勢GitMIT
- 玩轉 Ceph 的正確姿勢
- 開啟Git的正確姿勢Git
- laravel 使用 es 的正確姿勢Laravel
- java關流的正確姿勢Java
- 使用快取的正確姿勢快取
- Fragment commit 的正確姿勢FragmentMIT
- 基於Redis分散式鎖的正確開啟方式Redis分散式
- laradock 中安裝 Redis 擴充套件的正確姿勢Redis套件
- 原始碼|使用FutureTask的正確姿勢原始碼
- 相容iphone x劉海的正確姿勢iPhone
- npm run dev 的正確使用姿勢NPMdev
- 限制UITextField字數的正確姿勢UI
- 使用 Java 8 Optional 的正確姿勢Java
- MySQL 5.6建索引的正確姿勢MySql索引
- Python 操作 MySQL 的正確姿勢PythonMySql
- 演算法分析的正確姿勢演算法
- 解讀mysql的索引和事務的正確姿勢MySql索引
- Java日誌正確使用姿勢Java
- 解鎖多種JavaScript陣列去重姿勢JavaScript陣列
- canvas 影象旋轉與翻轉姿勢解鎖Canvas
- Pwn掉智慧手錶的正確姿勢
- Spring Boot使用AOP的正確姿勢Spring Boot
- Flutter 錯誤捕獲的正確姿勢Flutter
- 使用 react Context API 的正確姿勢ReactContextAPI