class Lock{
protected $redis;
protected $lockId;
public function __construct($redis){
$this->redis=$redis;
}
public function lock($k,$ttl=5,$retry=4,$usleep=10000){
$res=false;
while($retry-->0){
$v=session_create_id();
$res=$this->redis->set($k,$v,['NX','EX'=>$ttl]);
if($res){
$this->lockId[]=$v;
break;
}
usleep($usleep);
}
return $res;
}
public function unlock($k){
//能够删除自己的锁,不能删除别人的锁
if(isset($this->lockId[$k])){
$id=$this->lockId[$k];
$value=$this->redis->get($k]);
if($value==$id){
//sleep(5)//模拟网络延迟 ,这时候锁过期,另外一个请求加锁成功,下面代码就会删除掉另外一个请求的锁
return $this->redis->del($k);
}
上面极端情况下 可以出现误删别人锁的情况,因为Redis请求是有网络请求的
//改成
$script=<<<lua
if (redis.call('get',KEYS[1])==ARGV[1] ) then
return redis.call('del',KEYS[1])
end
return 0;
lua;
return $this->redis->eval($script,[$k,$id],1);
}
return false;
}
}

key 没有判断是不是自己的 会出现删除别人锁的情况

文章介绍了使用Redis实现的分布式锁类(classLock),包括构造函数、lock方法获取锁和unlock方法释放锁的过程。同时指出key未做身份验证可能导致误删他人锁的问题,以及为解决此问题引入Lua脚本进行安全解锁。
1010

被折叠的 条评论
为什么被折叠?



