解鎖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
程序變成:
網(wǎng)友評論