实现redis锁基于以下两个条件:
第一:
从2.6.12版本开始,redis为SET
命令增加了一系列选项:
EX
seconds – Set the specified expire time, in seconds.PX
milliseconds – Set the specified expire time, in milliseconds.NX
– Only set the key if it does not already exist.-
XX
– Only set the key if it already exist. EX
seconds – 设置键key的过期时间,单位时秒PX
milliseconds – 设置键key的过期时间,单位时毫秒NX
– 只有键key不存在的时候才会设置key的值XX
– 只有键key存在的时候才会设置key的值
第二:
基于redis的命令eval 运行 lua 命令保证原子性
首先我们看实例:
$redis = new Redis(); $redis->connect('127.0.0.1', 6379);
$token = uniqid("lock".rand(),true);$ttl = '5000';//NX代表KEY不存在时设置,相当于setNX//PX设置过期时间,单位是毫秒$res = $redis->set($lockKey, $token, ['NX', 'PX' => $ttl]);if($res){//获得了锁}//解锁$script = <<<LUASCRIPT if redis.call("GET", KEYS[1]) == ARGV[1] then return redis.call("DEL", KEYS[1]) else return 0 endLUASCRIPT;$redis->eval($script, [$lockKey, $token], 1);$lockKey = 'lock';
首先生成一个Token值,给redis加锁,返回true则就锁成功,否则则是其他进程获取了锁。
解锁的依赖redis lua 脚本的原子性,锁存在并且key值为token的时候才解锁。避免误解其他进程的锁。