使用Redis的 SETNX 命令可以实现分布式锁,下文介绍其实现方法。
SETNX命令简介
命令格式
SETNX key value
将 key 的值设为 value,当且仅当 key 不存在。
若给定的 key 已经存在,则 SETNX 不做任何动作。
SETNX 是SET if Not eXists的简写。
返回值
返回整数,具体为:
- 1,当 key 的值被设置
- 0,当 key 的值没被设置
例子:
redis> SETNX mykey “hello”
(integer) 1
redis> SETNX mykey “hello”
(integer) 0
redis> GET mykey
“hello”
redis>
使用SETNX实现分布式锁
多个进程执行以下Redis命令:
SETNX lock.foo <current Unix time + lock timeout + 1>
如果 SETNX 返回1,说明该进程获得锁,SETNX将键 lock.foo 的值设置为锁的超时时间(当前时间 + 锁的有效时间)。
如果 SETNX 返回0,说明其他进程已经获得了锁,进程不能进入临界区。进程可以在一个循环中不断地尝试 SETNX 操作,以获得锁。
上面的例子很容易产生死锁,并且 setnx和expire(设置超时时间)不是原子操作,当setnx后,redis宕掉,则此锁无法释放掉!
我们在redis官网看下 setnx命令的说明, 里面有一段描述
这里说明从2.6.12版本后, 就可以使用set来获取锁, Lua 脚本来释放锁
然后我们看下 set命令的说明 , 发现这里面可以有nx,xx等参数, 来实现 setnx 的功能.
而且这里再次提到了 获取锁和释放锁.
所以要多看官方文档!!!
所以,按照官方文档,我们获取锁就可以改版成这样
Jedis jedis = new Jedis("127.0.0.1", 6379);
String nx = jedis.set("nxxxx","ttt","NX","PX",1);
if ("OK".equals(nx)) {
System.out.println("我拿到了");
}
/* try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}*/
nx = jedis.set("nxxxx","ttt","NX","PX",1);
if ("OK".equals(nx)) {
System.out.println("ta拿到了");
}
输出结果为:
我拿到了
将上方注释打开:
输出的结果为:
我拿到了
ta拿到了
显然此锁起到了效果,而且此操作为原子操作,这种就可以达到setnx同事设置expire的效果
使用下发操作释放锁:
我们可以用下面这个Lua脚本来告诉Redis:删除这个key当且仅当这个key存在而且值是我期望的那个值。
if redis.call("get",KEYS[1]) == ARGV[1] then
return redis.call("del",KEYS[1])
else
return 0
end