主頁 > 知識(shí)庫 > 解鎖redis鎖的正確姿勢

解鎖redis鎖的正確姿勢

熱門標(biāo)簽:太原營銷外呼系統(tǒng) 小紅書怎么地圖標(biāo)注店 地圖標(biāo)注如何即時(shí)生效 竹間科技AI電銷機(jī)器人 最簡單的百度地圖標(biāo)注 地圖標(biāo)注費(fèi)用 玄武湖地圖標(biāo)注 西藏教育智能外呼系統(tǒng)價(jià)格 百度商家地圖標(biāo)注怎么做

解鎖redis鎖的正確姿勢

redis是php的好朋友,在php寫業(yè)務(wù)過程中,有時(shí)候會(huì)使用到鎖的概念,同時(shí)只能有一個(gè)人可以操作某個(gè)行為。這個(gè)時(shí)候我們就要用到鎖。鎖的方式有好幾種,php不能在內(nèi)存中用鎖,不能使用zookeeper加鎖,使用數(shù)據(jù)庫做鎖又消耗比較大,這個(gè)時(shí)候我們一般會(huì)選用redis做鎖機(jī)制。

setnx

鎖在redis中最簡單的數(shù)據(jù)結(jié)構(gòu)就是string。最早的時(shí)候,上鎖的操作一般使用setnx,這個(gè)命令是當(dāng):lock不存在的時(shí)候set一個(gè)val,或許你還會(huì)記得使用expire來增加鎖的過期,解鎖操作就是使用del命令,偽代碼如下:

if (Redis::setnx("my:lock", 1)) {
  Redis::expire("my:lock", 10);
  // ... do something

  Redis::del("my:lock")
}

這里其實(shí)是有問題的,問題就在于setnx和expire中間如果遇到crash等行為,可能這個(gè)lock就不會(huì)被釋放了。于是進(jìn)一步的優(yōu)化方案可能是在lock中存儲(chǔ)timestamp。判斷timestamp的長短。

set

現(xiàn)在官方建議直接使用set來實(shí)現(xiàn)鎖。我們可以使用set命令來替代setnx,就是下面這個(gè)樣子

if (Redis::set("my:lock", 1, "nx", "ex", 10)) {
  ... do something

  Redis::del("my:lock")
}

上面的代碼把my:lock設(shè)置為1,當(dāng)且僅當(dāng)這個(gè)lock不存在的時(shí)候,設(shè)置完成之后設(shè)置過期時(shí)間為10。

獲取鎖的機(jī)制是對了,但是刪除鎖的機(jī)制直接使用del是不對的。因?yàn)橛锌赡軐?dǎo)致誤刪別人的鎖的情況。

比如,這個(gè)鎖我上了10s,但是我處理的時(shí)間比10s更長,到了10s,這個(gè)鎖自動(dòng)過期了,被別人取走了,并且對它重新上鎖了。那么這個(gè)時(shí)候,我再調(diào)用Redis::del就是刪除別人建立的鎖了。

官方對解鎖的命令也有建議,建議使用lua腳本,先進(jìn)行g(shù)et,再進(jìn)行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是一個(gè)隨機(jī)數(shù),當(dāng)lock的時(shí)候,往redis的my:lock中存的是這個(gè)token,unlock的時(shí)候,先get一下lock中的token,如果和我要?jiǎng)h除的token是一致的,說明這個(gè)鎖是之前我set的,否則的話,說明這個(gè)鎖已經(jīng)過期,是別人set的,我就不應(yīng)該對它進(jìn)行任何操作。

所以:不要再使用setnx,直接使用set進(jìn)行鎖實(shí)現(xiàn)。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • 詳解Java如何實(shí)現(xiàn)基于Redis的分布式鎖
  • Redis實(shí)現(xiàn)分布式鎖的幾種方法總結(jié)
  • redis實(shí)現(xiàn)加鎖的幾種方法示例詳解
  • Redis上實(shí)現(xiàn)分布式鎖以提高性能的方案研究
  • Redis數(shù)據(jù)庫中實(shí)現(xiàn)分布式鎖的方法
  • php結(jié)合redis實(shí)現(xiàn)高并發(fā)下的搶購、秒殺功能的實(shí)例
  • Redis高并發(fā)問題的解決方法
  • Redis瞬時(shí)高并發(fā)秒殺方案總結(jié)
  • 如何利用Redis鎖解決高并發(fā)問題詳解

標(biāo)簽:林芝 贛州 廣東 澳門 景德鎮(zhèn) 揚(yáng)州 香港 唐山

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《解鎖redis鎖的正確姿勢》,本文關(guān)鍵詞  解鎖,redis,鎖,的,正確,姿勢,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《解鎖redis鎖的正確姿勢》相關(guān)的同類信息!
  • 本頁收集關(guān)于解鎖redis鎖的正確姿勢的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章