Redis分布式锁

104 阅读1分钟

1. 分布式锁主流的实现方案

  • 基于数据库实现分布式锁

  • 基于缓存(Redis等)

  • 基于Zookeeper

每一种分布式锁解决方案都有各自的优缺点:

性能:redis最高

可靠性:zookeeper最高

这里,我们就基于redis实现分布式锁。

2. 基本实现

Redis Setnx(SET if Not eXists) 命令在指定的 key 不存在时,为 key 设置指定的值。

语法

redis Setnx 命令基本语法如下:

redis 127.0.0.1:6379> SETNX KEY_NAME VALUE

返回值

设置成功,返回 1 。 设置失败,返回 0 。

实例

redis> EXISTS job                # job 不存在
(integer) 0
 
redis> SETNX job "programmer"    # job 设置成功
(integer) 1
 
redis> SETNX job "code-farmer"   # 尝试覆盖 job ,失败
(integer) 0
 
redis> GET job                   # 没有被覆盖
"programmer"

java代码

//获取分布式锁
public boolean getLock(String key, int expireTime) throws Exception {
        Jedis jedis = null;
        try {
                ......
                Long hasLock = jedis.setnx(key, "1");

                if(Objects.equals(1L, hasLock)) {
                        return true;
                }
        } finally {
                if (jedis != null) {
                        if(BooleanUtils.isTrue(jedis.exists(key))) {
                                jedis.expire(key, expireTime);
                        }
                        jedis.close();
                }
        }
        return false;
}

//释放分布式锁
public void freeLock(String key) throws Exception {
        Jedis jedis = null;
        try {
                ......
                if(BooleanUtils.isTrue(jedis.exists(key))) {
                        jedis.del(key);
                }
        } finally {
                if (jedis != null) {
                        jedis.close();
                }
        }
}

3. 注意事项

为了确保分布式锁可用,我们至少要确保锁的实现同时满足以下四个条件:

  • 互斥性。在任意时刻,只有一个客户端能持有锁。

  • 不会发生死锁。即使有一个客户端在持有锁的期间崩溃而没有主动解锁,也能保证后续其他客户端能加锁。

  • 解铃还须系铃人。加锁和解锁必须是同一个客户端,客户端自己不能把别人加的锁给解了。

  • 加锁和解锁必须具有原子性。