日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

解锁redis锁的正确姿势

發布時間:2024/1/17 编程问答 22 豆豆
生活随笔 收集整理的這篇文章主要介紹了 解锁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 somethingRedis::del("my:lock") }

這里其實是有問題的,問題就在于setnx和expire中間如果遇到crash等行為,可能這個lock就不會被釋放了。于是進一步的優化方案可能是在lock中存儲timestamp。判斷timestamp的長短。

set

現在官方建議直接使用set來實現鎖。我們可以使用set命令來替代setnx,就是下面這個樣子

if (Redis::set("my:lock", 1, "nx", "ex", 10)) {... do somethingRedis::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] thenreturn redis.call("del",KEYS[1]) elsereturn 0 end `return Redis::eval($script, "my:lock", $token) }if (lock()) {// do somethingunlock(); }

這里的token是一個隨機數,當lock的時候,往redis的my:lock中存的是這個token,unlock的時候,先get一下lock中的token,如果和我要刪除的token是一致的,說明這個鎖是之前我set的,否則的話,說明這個鎖已經過期,是別人set的,我就不應該對它進行任何操作。

所以:不要再使用setnx,直接使用set進行鎖實現。



本文轉自軒脈刃博客園博客,原文鏈接:http://www.cnblogs.com/yjf512/p/6597814.html,如需轉載請自行聯系原作者

總結

以上是生活随笔為你收集整理的解锁redis锁的正确姿势的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。