首頁  >  文章  >  後端開發  >  正確解redis鎖的方法

正確解redis鎖的方法

小云云
小云云原創
2017-12-14 13:35:582753瀏覽

redis是php的好朋友,在php寫業務過程中,有時候會使用到鎖的概念,同時只能有一個人可以操作某個行為。這時候我們就要用到鎖。鎖的方式有好幾種,php不能在記憶體中用鎖,不能使用zookeeper加鎖,使用資料庫做鎖又消耗比較大,這時候我們通常會選用redis做鎖機制。本文主要和大家分享正確解鎖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進行鎖定實作。

相關推薦:

php 使用redis鎖定限制並發存取類別

#php 使用redis鎖定限制並發存取類別範例

php操作redis的常用方法總結

#

以上是正確解redis鎖的方法的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn