【Redis】分布式锁基本理论与简单实现

目录

分布式锁

解释

  • 分布式锁是一种用于协调分布式系统中多个节点对共享资源进行访问的机制。
  • 在分布式系统中,多个节点可能同时竞争同一个资源,并且可能同时进行修改操作,这就会导致数据的不一致性和并发冲突的问题。
  • 为了解决这个问题,引入了分布式锁机制。

作用

  • 分布式锁可以确保在同一时刻只有一个节点能够对共享资源进行访问操作,其他节点需要等待该节点释放锁之后才能进行操作。
  • 分布式锁可以通过网络通信来实现,常见的实现方式有基于数据库的锁、基于缓存的锁、基于ZooKeeper的锁等。
  • 使用场景:分布式任务调度、分布式缓存、分布式事务等场景

特性

  1. 互斥性: 同一时刻只有一个节点能够获取到锁,其他节点需要等待。
  2. 可重入性: 同一个节点在获取到锁之后可以再次获取锁而不会被阻塞。
  3. 容错性: 锁的释放需要能够容忍节点的故障,确保锁能够被正常释放。
  4. 高性能: 分布式锁的实现需要保证高性能,避免成为系统的瓶颈。

实现方式

  1. 基于数据库:使用关系型数据库或者其他支持事务的数据库来实现分布式锁。可以通过在数据库中创建一个带有唯一索引的表或者行来确保只有一个进程能够成功获取锁。
  2. 基于文件系统:使用共享的文件系统来实现分布式锁。可以通过创建一个特定的文件来表示锁的状态,进程需要先创建文件或者尝试获得文件的独占写锁来获取锁。
  3. 基于ZooKeeper:使用ZooKeeper来实现分布式锁。可以通过在ZooKeeper中创建一个临时节点来表示锁的状态,只有创建成功的进程才能获取锁。
  4. 基于Redis:使用Redis的原子操作来实现分布式锁。可以通过在Redis中设置一个带有过期时间的键来表示锁的状态,只有成功设置锁的进程才能获取锁。

MySQL、Redis、Zookeeper三种方式对比

MySQL Redis Zookeeper
互斥 利用MySQL本身的互斥锁的机制 利用redis中setnx的互斥命令 利用节点的唯一性和有序性来实现互斥
高可用
高性能 一般 一般
安全性 断开连接,自动释放锁 利用锁超时时间。到期自动释放 临时节点,断开连接自动释放

原理

reids分布式锁原理

Redis分布式锁的原理基于Redis的单线程特性以及原子操作的特点。具体原理如下:

  1. 获取锁:当一个节点要获取分布式锁时,它会向Redis发送一个SETNX命令,将一个特定的键值对设置到Redis中。如果该键不存在,节点成功获取锁,并将该键值对设置为锁的持有者标识。如果该键已经存在,表示锁已经被其他节点持有,节点获取锁失败。

  2. 释放锁:当一个节点要释放分布式锁时,它会向Redis发送一个DEL命令,将该键值对从Redis中删除。只有持有锁的节点才能成功释放锁。

目的

  • 这样的实现基于Redis的SETNX命令的原子性保证,SETNX命令的语义是
    • 当键不存在时,设置键值对并返回1;
    • 当键已存在时,不设置值并返回0。
  • 通过SETNX命令的原子性,可以保证同一时刻只有一个节点能够成功获取锁。

容错

  • 为了防止分布式锁的死锁问题,可以为获取锁的操作设置一个过期时间。
  • 节点在获取锁的同时,可以为该键设置一个带有过期时间的键值对,确保即使节点在获取锁之后发生故障,如果过期时间到了,Redis也会自动释放该锁。
  • 为了提高分布式锁的可用性和容错性,还需要引入一些额外的机制,例如设置一个超时时间,避免长时间持有锁导致的问题。
  • 还可以使用分布式锁的续约机制,即在获取锁之后,定期向Redis发送续约命令,更新锁的过期时间,确保节点在持有锁的期间不会被自动释放。

redis简单分布式锁实现

锁接口

java 复制代码
public interface ILock {

    /**
     * 非阻塞方式,尝试获取锁
     * @param timeoutSec 锁持有的超时时间,过期后自动释放
     * @return true代表获取锁成功; false代表获取锁失败
     */
    boolean tryLock(long timeoutSec);

    /**
     * 释放锁,有加锁就要有释放锁
     */
    void unlock();
}

实现类

java 复制代码
public class SimpleRedisLock implements ILock {

	// 业务名称
    private String name;
    private StringRedisTemplate stringRedisTemplate;

	// 通过构造方法将name和stringRedisTemplate传入
    public SimpleRedisLock(String name, StringRedisTemplate stringRedisTemplate) {
        this.name = name;
        this.stringRedisTemplate = stringRedisTemplate;
    }

    private static final String KEY_PREFIX = "lock:";

    @Override
    public boolean tryLock(long timeoutSec) {
        // 获取线程标识
        long threadId = Thread.currentThread().getId();
        // 获取锁
        Boolean success = stringRedisTemplate.opsForValue()
                .setIfAbsent(KEY_PREFIX + name, threadId + "", timeoutSec, TimeUnit.SECONDS);
        return Boolean.TRUE.equals(success);
    }

    @Override
    public void unlock() {
        //通过del删除锁
        stringRedisTemplate.delete(KEY_PREFIX + name);
    }
}

下单场景的实现

java 复制代码
// 使用Redis分布式锁
// 创建锁对象
SimpleRedisLock lock = new SimpleRedisLock("order:" + userId, stringRedisTemplate);
// 获取锁对象
boolean isLock = lock.tryLock(5);
// 加锁失败
if (!isLock) {
   return Result.fail("不允许重复下单");
}
try {
   // 获取代理对象(事务)
   IVoucherOrderService proxy = (IVoucherOrderService) AopContext.currentProxy();
   return proxy.createVoucherOrder(voucherId);
} finally {
   // 释放锁
   lock.unlock();
}

容错场景1

  1. 线程1先获取锁后,由于业务阻塞还没执行完成,线程1的锁超时后自动释放
  2. 线程2在线程1的锁超时自动释放后,进行加锁成功
  3. 正好线程1将业务接着执行完后,需要释放锁,此时释放的就是线程2的锁,造成了误删问题
  4. 误删后,线程3又加锁成功,此时,线程2和线程3就出现了并发执行业务,造成并发安全问题
解决思路
  • 在获取锁时:存入线程标识,比如可以用UUID这类的唯一序列
  • 在释放锁时:先获取锁中的线程标识,判断是否与当前线程标识一致
    • 如果一致则释放锁
    • 如果不一致则不释放锁
  • 不要直接将线程id作为线程标识,因为不同JVM中的线程id可能一样,所以可以用 线程id+UUID 作为线程标识
优化代码
java 复制代码
public class SimpleRedisLock implements ILock {

    // 业务名称
    private String name;
    private StringRedisTemplate stringRedisTemplate;

    // 通过构造方法将name和stringRedisTemplate传入
    public SimpleRedisLock(String name, StringRedisTemplate stringRedisTemplate) {
        this.name = name;
        this.stringRedisTemplate = stringRedisTemplate;
    }

    private static final String KEY_PREFIX = "lock:";
    private static final String ID_PREFIX = UUID.randomUUID().toString(true) + "-";

    @Override
    public boolean tryLock(long timeoutSec) {
        // 获取线程标识
        String threadId = ID_PREFIX + Thread.currentThread().getId();
        // 获取锁
        Boolean success = stringRedisTemplate.opsForValue()
                .setIfAbsent(KEY_PREFIX + name, threadId, timeoutSec, TimeUnit.SECONDS);
        return Boolean.TRUE.equals(success);
    }

    @Override
    public void unlock() {
        // 获取线程标识
        String threadId = ID_PREFIX + Thread.currentThread().getId();
        // 获取锁中的标识
        String id = stringRedisTemplate.opsForValue().get(KEY_PREFIX + name);
        // 判断标识是否一致
        if(threadId.equals(id)) {
            // 释放锁
            stringRedisTemplate.delete(KEY_PREFIX + name);
        }
    }
}

容错场景2

  1. 线程1执行完业务后,准备释放锁
  2. 先判断完锁一致后,正准备释放时,发生了阻塞(例如:GC时所有线程会阻塞),恰好线程1在阻塞期间,锁超时被释放
  3. 线程2获取锁成功,此时线程1被唤醒后,继续释放锁,由于之前判断过锁的标识,所以直接释放锁,但是此时的锁是线程2的
  4. 线程3又加锁成功,此时,线程2和线程3就出现了并发执行业务,造成并发安全问题
Lua脚本
  • Lua脚本是一种轻量级的编程语言,用于嵌入式系统和游戏开发中。其设计目标是为了简单、可扩展和快速。
  • Lua脚本具有简洁的语法和功能强大的特性,包括动态类型、自动内存管理和高阶函数支持。它可以被嵌入到其他程序中,以提供脚本化的功能。由于其轻量级和高性能的特点,Lua脚本被广泛应用于游戏脚本、应用程序的扩展和配置文件等方面。
  • Lua脚本可以通过与其他编程语言的接口交互,例如C、C++和Java,使开发人员可以在应用程序中使用Lua脚本来实现灵活的功能和逻辑。此外,Lua还具有丰富的标准库和大量的第三方库,使开发人员能够快速开发出各种类型的应用程序。
Redis利用Lua脚本解决多条命令原子性问题
  • Redis提供了Lua脚本功能,在一个脚本中编写多条Redis命令,确保多条命令执行时的原子性

    powershell 复制代码
    # 执行Redis命令
    redis.call('命令名称', 'key', '其他参数', ...)
  • 例如,我们要先执行set name zhangsan,再执行get name,则脚本如下:

    powershell 复制代码
    # 先执行 set name zhangsan
    redis.call('set', 'name', 'zhangsan')
    # 再执行 get name
    local name = redis.call('get', 'name')
    # 返回
    return name
  • 写好脚本以后,需要用Redis命令来调用脚本,例如,我们要执行 redis.call('set', 'name', 'jack') 这个脚本,语法如下:

    • 双引号内表示脚本内容

    • 最后的0表示脚本需要的key类型的参数个数

      powershell 复制代码
      EVAL "return redis.call('set','name','zhangsan')" 0
  • 如果脚本中的key、value不想写死,可以作为参数传递。key类型参数会放入KEYS数组,其它参数会放入ARGV数组,在脚本中可以从KEYS和ARGV数组获取这些参数:

    • name传给KEYS[1]

    • zhangsan传给ARGV[1]

      powershell 复制代码
      EVAL "return redis.call('set',KEYS[1],ARGV[1])" 1 name zhangsan
释放锁的业务流程
  1. 获取锁中的线程标识
  2. 判断是否与指定的标识(当前线程标识)一致
    • 如果一致则释放锁(删除)
    • 如果不一致则什么都不做
Lua脚本来表示
powershell 复制代码
-- 这里的 KEYS[1] 就是锁的key,这里的ARGV[1] 就是当前线程标识
-- 获取锁中的标识,判断是否与当前线程标识一致
if (redis.call('GET', KEYS[1]) == ARGV[1]) then
  -- 一致,则删除锁
  return redis.call('DEL', KEYS[1])
end
-- 不一致,则直接返回
return 0
优化代码
  • 基于Lua脚本实现分布式锁的释放锁逻辑
  • RedisTemplate调用Lua脚本的API如下:
java 复制代码
public class SimpleRedisLock implements ILock {

    // 业务名称
    private String name;
    private StringRedisTemplate stringRedisTemplate;

    // 通过构造方法将name和stringRedisTemplate传入
    public SimpleRedisLock(String name, StringRedisTemplate stringRedisTemplate) {
        this.name = name;
        this.stringRedisTemplate = stringRedisTemplate;
    }

    private static final String KEY_PREFIX = "lock:";
    private static final String ID_PREFIX = UUID.randomUUID().toString(true) + "-";
    // 加载Lua脚本
    private static final DefaultRedisScript<Long> UNLOCK_SCRIPT;
    static {
        UNLOCK_SCRIPT = new DefaultRedisScript<>();
        //将编写的Lua脚本放在resources目录下,比如名称为:unlock.lua
        UNLOCK_SCRIPT.setLocation(new ClassPathResource("unlock.lua"));
        UNLOCK_SCRIPT.setResultType(Long.class);
    }

    @Override
    public boolean tryLock(long timeoutSec) {
        // 获取线程标识
        String threadId = ID_PREFIX + Thread.currentThread().getId();
        // 获取锁
        Boolean success = stringRedisTemplate.opsForValue()
                .setIfAbsent(KEY_PREFIX + name, threadId, timeoutSec, TimeUnit.SECONDS);
        return Boolean.TRUE.equals(success);
    }

    @Override
    public void unlockL() {
        // 调用Lua脚本
        stringRedisTemplate.execute(
                UNLOCK_SCRIPT,
                Collections.singletonList(KEY_PREFIX + name),
                ID_PREFIX + Thread.currentThread().getId());
    }

总结

  • Redis的分布式锁实现其实就是利用setnx/setex获取锁,并设置过期时间,保存线程标识

  • 释放锁时先判断线程标识是否与自己一致,一致则删除锁

  • Redis的分布式的优点:

    • 利用setnx满足互斥性
    • 利用setex保证故障时锁依然能释放,避免死锁,提高安全性
    • 利用Redis集群保证高可用和高并发特性
相关推荐
2401_857622665 分钟前
SQL Server的隐私盾牌:动态数据屏蔽(DMS)全面解析
数据库·oracle
逸风扬8 分钟前
被裁了(9年)
java·程序人生
DKPT15 分钟前
如何确定MySQL中哪些列适合做索引
数据库·spring boot·spring·spring cloud·缓存
AskHarries26 分钟前
Spring Boot集成fastjson2快速入门Demo
java·spring boot·fastjson
极客先躯27 分钟前
中级java每日一道面试题-2024年7月3日
java·开发语言·java每日一道面试题
lakernote28 分钟前
如何系列 如何确保Kafka消息可靠性/防止消息丢失
分布式·kafka·linq
wilsonzane33 分钟前
Mongodb集群中的分布式读写
数据库·分布式·mongodb
关中雪41 分钟前
【应届应知应会】SQL常用知识点50道
数据库·mysql·nosql·秋招·校招·春招
AskHarries44 分钟前
Spring Boot集成geode快速入门Demo
java·spring boot·后端·geode
zxrhhm1 小时前
MySQL中使用PROFILING来查看SQL执行流程
数据库·mysql