解鎖redis鎖的正確姿勢
redis是php的好朋友,在php寫業(yè)務(wù)過程中,有時候會使用到鎖的概念,同時只能有一個人可以操作某個行為。這個時候我們就要用到鎖。鎖的方式有好幾種,php不能在內(nèi)存中用鎖,不能使用zookeeper加鎖,使用數(shù)據(jù)庫做鎖又消耗比較大,這個時候我們一般會選用redis做鎖機制。
setnx
鎖在redis中最簡單的數(shù)據(jù)結(jié)構(gòu)就是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就不會被釋放了。于是進一步的優(yōu)化方案可能是在lock中存儲timestamp。判斷timestamp的長短。
set
現(xiàn)在官方建議直接使用set來實現(xiàn)鎖。我們可以使用set命令來替代setnx,就是下面這