百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

聊一聊业务中Redis锁的实现 redission锁原理

wptr33 2024-12-17 16:46 19 浏览

背景

随着业务的发展,IT项目逐渐演进为微服务架构,这也带来了一些挑战,例如在锁的使用方面。在传统的单体应用中,锁通常在整个应用程序中共享,然而在微服务架构中,每个服务都有独立的数据库和缓存,这意味着锁需要在服务之间进行协调。

以下通过两张图来阐明本地锁和分布式锁的区别:

image

image

为了应对这一问题,分布式锁应运而生,而其中最常采用的技术之一是基于 Redis 实现的分布式锁。

基本实现思路

通过 Redis 的 SET NX 命令,我们可以实现一种原子操作:只有在指定的 key 不存在时,写入才会成功;若 key 已存在,则写入会失败。

public synchronized boolean tryLock() {
    if (this.locking) {
        log.warn("【Redis锁异常】key=[{}] 重复请求锁:不支持重入,请检查代码", this.key);
        return false;
    }


    //尝试拿锁,如果拿不到就等待并重试,最多等待this.maxWaitSeconds
    boolean success = tryWaitForLock();

    if (success) {
        //已获取到锁
        this.locking = true;
        //注册到manager,以进行续期管理
        manager.registerLock(this);
        lastRenewalTime = LocalDateTime.now();
        return true;
    } else {
        //未获取到锁
        log.warn("【Redis锁获取失败】key=[{}] 取锁失败,且等待时间超出最多等待[{}]秒 value=[{}]", this.key, this.maxWaitSeconds, this.value);
        return false;
    }

}

public boolean tryWaitForLock(String lockKey, String lockValue, long expireTime, long maxWaitSeconds) {
        final int sleepMills = 100;
        final long maxWaitMills = maxWaitSeconds * 1000;
        final long maxLoop = maxWaitMills / sleepMills;
        final Random random = new Random();
        for (int idx = 0; idx <= maxLoop; idx++) {
            try {
                String result = redisCache.set(lockKey, lockValue, NX, EX, expireTime);
                if (LOCK_SUCCESS.equalsIgnoreCase(result)) {
                    return true;
                }
                if (idx < maxLoop) {
                    // 20ms上下浮动,避免波峰
                    Thread.sleep(sleepMills + (20 - random.nextInt(40)));
                    log.debug("【等待Redis锁】key=[{}] 已等待[{}]毫秒 maxWaitMills=[{}]毫秒 value=[{}]", lockKey,
                            (idx + 1) * sleepMills, maxWaitMills, lockValue);
                }
            } catch (Exception e) {
                log.debug("【等待Redis锁】key=[{}] 等待时出现异常 已等待[{}]毫秒 maxWaitMills=[{}]毫秒  value=[{}]", lockKey,
                        (idx + 1) * sleepMills, maxWaitMills, lockValue, e);
            }
        }
        return false;
    }

在获取锁的过程中,如果第一次尝试失败,会进行多次尝试,若依然无法获取锁,则返回失败。

然而,我们仍需处理一种情况:业务执行时间较长,但锁已过期。为应对这种情况,客户端可以在成功设置锁后,启动定时任务,在锁即将超时之前更新锁的超时时间,以确保业务完成的同时保持锁的有效性。

private RedisLockRenewalManager() {
    ScheduledExecutorService executorService = Executors.newScheduledThreadPool(
            3,
            runnable -> new Thread(runnable, "redis-lock-renewal")
    );
    executorService.scheduleAtFixedRate(this::heartbeat, 500, 500, TimeUnit.MILLISECONDS);
}

private void heartbeat() {
    for (RedisLock lock : locks) {
        if (lock.isLocking()) {
            //正在锁定中的,检查是否需要续期,需要的自动执行续期
            lock.renewal();
        } else {
            //移除未使用的锁,避免内存泄露
            locks.remove(lock);
        }
    }
}

public void renewal() {
    LocalDateTime nextRenewalTime = lastRenewalTime.plusSeconds(this.expireSeconds / 2);
    if (nextRenewalTime.isAfter(LocalDateTime.now())) {
        log.trace("【redis锁续期】key=[{}] 过期时间[{}s]未过半,暂不需要刷新,本次跳过", this.key, this.expireSeconds);
        return;
    }
    Object result = redisCache.eval(RENEWAL_LUA_SCRIPT, 1, this.key, this.value, String.valueOf(this.expireSeconds));
    if (Objects.nonNull(result) && Objects.equals(result, 1L)) {
        this.lastRenewalTime = LocalDateTime.now();
        log.debug("【redis锁续期成功】key=[{}]", this.key);
    } else {
        String redisValue = redisCache.get(this.key);
        log.warn("【redis锁续期失败】key=[{}] this.value=[{}] redis.value=[{}]", this.key, this.value, redisValue);
    }
}

于是加锁的整个过程如图:

image

加锁环节几个问题解决了,锁释放应如何实现呢?
可以使用redis的
del命令对锁进行释放,这里释放的时候需要判断当前的锁对象是不是自己的,避免误释放了。因此也采用Redis脚本命令的方式:

/**
 * 解锁lua脚本
 */
public static final String UNLOCK_LUA_SCRIPT = "if (redis.call('GET', KEYS[1]) == ARGV[1]) then return redis.call('DEL', KEYS[1]) else return 0 end";

/**
 * 续期lua脚本
 */
public static final String RENEWAL_LUA_SCRIPT = "if redis.call('GET', KEYS[1]) == ARGV[1] then return redis.call('EXPIRE', KEYS[1], ARGV[2]) else return 0 end";

public boolean unlock() {
    if (!this.locking) {
        log.warn("【Redis锁异常】key=[{}] 重复解锁,请检查代码", this.key);
        return false;
    }
    try {
        Object result = redisCache.eval(UNLOCK_LUA_SCRIPT, 1, this.key, this.value);
        if (Objects.isNull(result) || !Objects.equals(1L, result)) {
            String redisValue = redisCache.get(this.key);
            log.warn("【释放redis锁失败】key=[{}] this.value=[{}] redis.value=[{}]", this.key, this.value, redisValue);
        }
    }catch (Exception exception){
        log.warn("【释放Redis锁异常】key=[{}],msg=[{}]",this.key,exception.getMessage(),exception);
    }finally {
        // 无论释放锁实际是否成功,均返回成功。
        // 如果释放锁失败,则由redis自动过期清除该锁,需要自动禁止续期
        this.locking = false;
        manager.unregisterLock(this);
    }
    return true;
}

这里对key的定义是这样的:

private String generateValue() {
    Thread thread = Thread.currentThread();
    String hostname = System.getProperty("HOSTNAME");
    return UUID.randomUUID() + "#34; + thread.getName() + "#" + thread.getId() + "@" + hostname;
}

在分布式环境下,需要将机器名也作为key的一部分,避免UUID在多机器上出现重复的问题(虽然是小概率)。

最终使用的代码如下:

public void demo() {

    RedisCache redisCache = createRedisCache();
    RedisLockFactory lockFactory = new RedisLockFactory(redisCache);

    //分布式锁使用参考模板
    String lockKey = "test-lock-key";
    RedisLock redisLock = lockFactory.create(lockKey, 10, 10);
    try {
        boolean success = redisLock.tryLock();
        if (!success) {
            log.warn("【业务流程名】Redis锁[{}]申请失败", lockKey);
            return;
        }
        //region 业务处理代码
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        //endregion
    } finally {
        if (redisLock.isLocking() && !redisLock.unlock()) {
            log.warn("【业务流程名】Redis锁[{}]释放失败", lockKey);
        }
    }
}

总结

本文通过几个代码示例详细介绍了 Redis 分布式锁的几个重要特性:

  • 互斥性:在任意时刻,只允许一个客户端持有锁,确保了锁的独占性。
  • 无死锁:即使在某个客户端持有锁的期间发生崩溃,未主动解锁的情况下,也能确保后续其他客户端能够正常加锁,避免了死锁问题。
  • 自持自解:加锁和解锁必须由同一客户端(线程)完成,禁止客户端解除其他客户端持有的锁,确保了锁的所有权。

除了这些优点,我们也需要注意以下问题:

  • 主从切换问题:在单实例环境中,该分布式锁方案是可行的。然而,在 Redis 集群环境中,尤其是在主从切换时,可能会出现问题。例如,当主节点挂掉,从节点升级为主节点,但数据尚未完全同步时,新的主节点上的锁信息可能会丢失,导致后续请求获取了无效的锁。为解决这一问题,可考虑使用 Redisson 框架的 Redlock 算法。
  • 不支持重入:该分布式锁不支持重入,对于某些场景需要自己调用自己的递归调用可能会出现问题。为解决这一限制,可以参考 AQS 实现,对锁进行计数,每进入一次加1,每释放一次减1,数量为0时释放锁,实现了对锁的可重入性。

号外号外

总结了很多年的Java面试宝典,相关的高频面试点,全是大厂真题,并免费提供面试问题咨询、一对一简历优化模拟面试~

地址:https://github.com/xbox1994/Java-Interview

相关推荐

Python自动化脚本应用与示例(python办公自动化脚本)

Python是编写自动化脚本的绝佳选择,因其语法简洁、库丰富且跨平台兼容性强。以下是Python自动化脚本的常见应用场景及示例,帮助你快速上手:一、常见自动化场景文件与目录操作...

Python文件操作常用库高级应用教程

本文是在前面《Python文件操作常用库使用教程》的基础上,进一步学习Python文件操作库的高级应用。一、高级文件系统监控1.1watchdog库-实时文件系统监控安装与基本使用:...

Python办公自动化系列篇之六:文件系统与操作系统任务

作为高效办公自动化领域的主流编程语言,Python凭借其优雅的语法结构、完善的技术生态及成熟的第三方工具库集合,已成为企业数字化转型过程中提升运营效率的理想选择。该语言在结构化数据处理、自动化文档生成...

14《Python 办公自动化教程》os 模块操作文件与文件夹

在日常工作中,我们经常会和文件、文件夹打交道,比如将服务器上指定目录下文件进行归档,或将爬虫爬取的数据根据时间创建对应的文件夹/文件,如果这些还依靠手动来进行操作,无疑是费时费力的,这时候Pyt...

python中os模块详解(python os.path模块)

os模块是Python标准库中的一个模块,它提供了与操作系统交互的方法。使用os模块可以方便地执行许多常见的系统任务,如文件和目录操作、进程管理、环境变量管理等。下面是os模块中一些常用的函数和方法:...

21-Python-文件操作(python文件的操作步骤)

在Python中,文件操作是非常重要的一部分,它允许我们读取、写入和修改文件。下面将详细讲解Python文件操作的各个方面,并给出相应的示例。1-打开文件...

轻松玩转Python文件操作:移动、删除

哈喽,大家好,我是木头左!Python文件操作基础在处理计算机文件时,经常需要执行如移动和删除等基本操作。Python提供了一些内置的库来帮助完成这些任务,其中最常用的就是os模块和shutil模块。...

Python 初学者练习:删除文件和文件夹

在本教程中,你将学习如何在Python中删除文件和文件夹。使用os.remove()函数删除文件...

引人遐想,用 Python 获取你想要的“某个人”摄像头照片

仅用来学习,希望给你们有提供到学习上的作用。1.安装库需要安装python3.5以上版本,在官网下载即可。然后安装库opencv-python,安装方式为打开终端输入命令行。...

Python如何使用临时文件和目录(python目录下文件)

在某些项目中,有时候会有大量的临时数据,比如各种日志,这时候我们要做数据分析,并把最后的结果储存起来,这些大量的临时数据如果常驻内存,将消耗大量内存资源,我们可以使用临时文件,存储这些临时数据。使用标...

Linux 下海量文件删除方法效率对比,最慢的竟然是 rm

Linux下海量文件删除方法效率对比,本次参赛选手一共6位,分别是:rm、find、findwithdelete、rsync、Python、Perl.首先建立50万个文件$testfor...

Python 开发工程师必会的 5 个系统命令操作库

当我们需要编写自动化脚本、部署工具、监控程序时,熟练操作系统命令几乎是必备技能。今天就来聊聊我在实际项目中高频使用的5个系统命令操作库,这些可都是能让你效率翻倍的"瑞士军刀"。一...

Python常用文件操作库使用详解(python文件操作选项)

Python生态系统提供了丰富的文件操作库,可以处理各种复杂的文件操作需求。本教程将介绍Python中最常用的文件操作库及其实际应用。一、标准库核心模块1.1os模块-操作系统接口主要功能...

11. 文件与IO操作(文件io和网络io)

本章深入探讨Go语言文件处理与IO操作的核心技术,结合高性能实践与安全规范,提供企业级解决方案。11.1文件读写11.1.1基础操作...

Python os模块的20个应用实例(python中 import os模块用法)

在Python中,...