Redisson常用方法

Redisson

参考: 原文链接

定义:Redisson 是一个用于与 Redis 进行交互的 Java 客户端库

优点:很多

1. 入门

1.1 安装

<!--redission-->
<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson</artifactId>
    <version>3.31.0</version>
</dependency>

<!--starter-->
<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson-spring-boot-starter</artifactId>
    <version>3.18.0</version>
</dependency>

1.2 配置

@Configuration
public class RedissonConfig {

    @Bean
    public RedissonClient redissonClient(){
        // 配置
        Config config = new Config();
        config.useSingleServer().setAddress("redis://192.168.133.136:6379")
            .setPassword("111111");
        // 创建RedissonClient对象
        return Redisson.create(config);
    }
}
---------------------------------使用yml配置-----------------------------------------
redisson:
  singleServerConfig:
    address: "redis://192.168.133.136:6379"  # Redis 服务器地址
    password: "111111"  # 如果有密码,填入密码
    connectionMinimumIdleSize: 10  # 最小空闲连接数
    connectionPoolSize: 64  # 连接池大小
    idleConnectionTimeout: 10000  # 空闲连接最大存活时间
    connectTimeout: 10000  # 连接超时
    timeout: 10000  # 请求超时
    retryAttempts: 3  # 重试次数
    retryInterval: 1500  # 重试间隔(毫秒)

1.3 使用

@Autowired
private RedissonClient redissonClient;

@Test
void testRedisson() throws Exception {
    //获取锁(可重入),指定锁的名称
    RLock lock = redissonClient.getLock("sanjin");
    //尝试获取锁,参数分别是:获取锁的最大等待时间(期间会重试),锁自动释放时间,时间单位
    boolean isLock = lock.tryLock(1, 10, TimeUnit.SECONDS);
    //判断获取锁成功
    if (isLock) {
        try {
            System.out.println("执行业务");
        } finally {
            //释放锁
            lock.unlock();
        }
    }
}
----------------结果------------------
执行业务

2. 可重入锁原理

2.1 加锁

这是可重入锁接口,

public interface RLock extends Lock, RLockAsync {
    String getName();

    void lockInterruptibly(long var1, TimeUnit var3) throws InterruptedException;

    // 有等待时间
    boolean tryLock(long var1, long var3, TimeUnit var5) throws InterruptedException;
	// 无等待时间
    void lock(long var1, TimeUnit var3);

    boolean forceUnlock();

    boolean isLocked();

    boolean isHeldByThread(long var1);

    boolean isHeldByCurrentThread();

    int getHoldCount();

    long remainTimeToLive();
}

看一下lock实现方法, 明白大体逻辑即可

  1. 如果成功,立即返回,如果失败,订阅锁的释放事件
  2. 在锁释放时,重新尝试获取锁,如果仍未成功(又被抢了),根据 TTL 再次等待,直到获取锁成功
  3. 在方法退出前,取消对锁释放事件的订阅,避免资源浪费
private void lock(long leaseTime, TimeUnit unit, boolean interruptibly) throws InterruptedException {
    // 保证只有锁的持有线程可以释放锁
    long threadId = Thread.currentThread().getId();
    // -1通常表示锁的等待时间设置为无限制(立即尝试获取锁)
    // 如果返回 null,表示成功获取到锁, ttl表示锁的剩余过期时间
    Long ttl = this.tryAcquire(-1L, leaseTime, unit, threadId);
    
    if (ttl != null) {
       	// 如果未获取到锁,订阅锁的释放事件
        CompletableFuture<RedissonLockEntry> future = this.subscribe(threadId);
        // 设置订阅超时,防止由于网络或其他问题导致的长时间等待
        this.pubSub.timeout(future);
        RedissonLockEntry entry;
        // 阻塞等待订阅结果
        if (interruptibly) {
            entry = (RedissonLockEntry)this.commandExecutor.getInterrupted(future);
        } else {
            entry = (RedissonLockEntry)this.commandExecutor.get(future);
        }
        try {
            // 进入一个无限循环,不断尝试重新获取锁,直到成功为止
            while(true) {
                // 再次尝试获取锁
                ttl = this.tryAcquire(-1L, leaseTime, unit, threadId);
                // 如果返回 null,表示成功获取锁,直接退出方法
                if (ttl == null) {
                    return;
                }
                // 如果返回一个非 null 的值 ttl,表示锁仍被占用,需要根据剩余时间等待
                if (ttl >= 0L) {
                    try {
                        // 使用计数器(CountDownLatch 的一种实现)来等待锁的释放通知
                        entry.getLatch().tryAcquire(ttl, TimeUnit.MILLISECONDS);
                    } catch (InterruptedException var14) {
                        if (interruptibly) {
                            throw var14;
                        }
                        entry.getLatch().tryAcquire(ttl, TimeUnit.MILLISECONDS);
                    }
                // 如果 TTL 为负数(可能表示锁的持有者未设置自动过期时间),线程将等待一个释放通知
                } else if (interruptibly) {
                    entry.getLatch().acquire();
                } else {
                    entry.getLatch().acquireUninterruptibly();
                }
            }
        } finally {
            // 无论锁是否成功获取,最终都会释放订阅,以避免资源泄漏
            this.unsubscribe(entry, threadId);
        }
    }
}

具体加锁逻辑tryAcquire

  • KEYS[1]:锁的 Redis 键,通常为锁的唯一标识
  • ARGV[1]:锁的过期时间(毫秒)
  • ARGV[2]:当前线程的唯一标识(value)
<T> RFuture<T> tryLockInnerAsync(long waitTime, long leaseTime, TimeUnit unit, long threadId, RedisStrictCommand<T> command) {
    
    // 异步执行 Redis 的 EVAL 命令
    return this.evalWriteAsync(this.getRawName(), LongCodec.INSTANCE, command,
		// 判断锁是否已存在,如果锁不存在,创建锁并设置过期时间
        "if (redis.call('exists', KEYS[1]) == 0) then " +
		// 使用 HINCRBY 创建锁并设置当前线程的持有次数为 1,利用hash结构
        "redis.call('hincrby', KEYS[1], ARGV[2], 1); " +
		// 设置锁的过期时间,确保锁在持有者崩溃后释放
        "redis.call('pexpire', KEYS[1], ARGV[1]); " +
		// 返回 nil 表示锁已成功获取
        "return nil; " +
        "end; " +
		// 锁已被当前线程持有的情况, 判断锁是否被当前线程持有
        "if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) then " +
		// 增加持有次数(支持可重入)
        "redis.call('hincrby', KEYS[1], ARGV[2], 1); " +
		// 更新锁的过期时间
        "redis.call('pexpire', KEYS[1], ARGV[1]); " +
        "return nil; " +
        "end; " +
		// 返回锁的剩余过期时间(毫秒),用于通知调用者等待或重试
        "return redis.call('pttl', KEYS[1]);",
        Collections.singletonList(this.getRawName()), 
		new Object[]{unit.toMillis(leaseTime), 
		this.getLockName(threadId)});
    
}

具体加锁逻辑如下图

2.2 续锁

主要就是根据leaseTime判断如何操作

指定了leaseTime:设置过期时间为leaseTime,不启用看门狗

不指定leaseTime:设置默认过期时间(30s),并且启用看门狗

指不指定是看 lock.lock() 是否传入了值

private <T> RFuture<Long> tryAcquireAsync(long waitTime, long leaseTime, TimeUnit unit, long threadId) {
    RFuture ttlRemainingFuture;
    // 尝试获取锁
    if (leaseTime > 0L) {
        // 使用指定的过期时间尝试获取锁, 适合短期锁场景,过期后无需续期
        ttlRemainingFuture = this.tryLockInnerAsync(waitTime, leaseTime, unit, threadId, RedisCommands.EVAL_LONG);
    } else {
        // 使用内部默认的锁租约时间 internalLockLeaseTime(30秒), 适合长期锁场景,通常需要续期机制
        ttlRemainingFuture = this.tryLockInnerAsync(waitTime, this.internalLockLeaseTime, TimeUnit.MILLISECONDS, threadId, RedisCommands.EVAL_LONG);
    }
    
    // 处理锁获取结果
    CompletionStage<Long> f = ttlRemainingFuture.thenApply((ttlRemaining) -> {
        // 如果 ttlRemaining == null,说明锁成功获取
        if (ttlRemaining == null) {
            if (leaseTime > 0L) {
                // 将 internalLockLeaseTime 更新为指定的租约时间,后续 Redis 锁命令会使用该值设置锁的过期时间
                this.internalLockLeaseTime = unit.toMillis(leaseTime);
            } else {
                // 看门狗机制续租
                // 调用 scheduleExpirationRenewal 方法,开启后台续期任务,确保锁不会因过期时间耗尽而释放
                this.scheduleExpirationRenewal(threadId);
            }
        }
        // 表示锁已被其他线程持有,返回锁的剩余有效时间
        return ttlRemaining;
    });
    return new CompletableFutureWrapper(f);
}

2.3 解锁

最外层的解锁方法

public void unlock() {
    try {
        // 异步调用解锁方法,并等待其执行完成
        this.get(this.unlockAsync(Thread.currentThread().getId()));
    } catch (RedisException var2) {
        if (var2.getCause() instanceof IllegalMonitorStateException) {
            throw (IllegalMonitorStateException)var2.getCause();
        } else {
            throw var2;
        }
    }
}

解锁并处理解锁后的步骤(取消看门狗机制…)

public RFuture<Void> unlockAsync(long threadId) {
    // 异步调用解锁方法
    RFuture<Boolean> future = this.unlockInnerAsync(threadId);
    // 使用 handle() 方法处理解锁结果和异常
    CompletionStage<Void> f = future.handle((opStatus, e) -> {
        // 取消锁的续期
        this.cancelExpirationRenewal(threadId);
        // 如果异步操作抛出异常,包装并抛出 CompletionException
        if (e != null) {
            throw new CompletionException(e);
        } else if (opStatus == null) {
            // 如果解锁操作状态为 null,说明当前线程未持有锁,抛出 IllegalMonitorStateException
            IllegalMonitorStateException cause = new IllegalMonitorStateException("attempt to unlock lock, not locked by current thread by node id: " + this.id + " thread-id: " + threadId);
            throw new CompletionException(cause);
        } else {
            // 如果操作成功,则返回 null
            return null;
        }
    });
    return new CompletableFutureWrapper(f);
}

具体解锁的redis执行步骤

参数:

  • KEYS[1]:锁的键名。
  • KEYS[2]:用于发布解锁事件的频道名。
  • ARGV[1]:解锁事件消息。
  • ARGV[2]:锁的过期时间。
  • ARGV[3]:锁的名称(用于计数器)。

步骤:

  1. 检查当前线程是否持有锁

  2. 减少锁计数器

  3. 如果计数器值大于零,更新锁的过期时间

  4. 如果计数器值为零,删除锁并发布解锁事件

protected RFuture<Boolean> unlockInnerAsync(long threadId) {
	return this.evalWriteAsync(this.getRawName(), LongCodec.INSTANCE, RedisCommands.EVAL_BOOLEAN,
		// 检查指定的锁是否存在。如果不存在(0),返回 nil,表示当前线程没有持有锁。
        "if (redis.call('hexists', KEYS[1], ARGV[3]) == 0) then " +
        "return nil;" +
        "end;" +
		// 将锁的计数器减一。如果当前线程持有锁,这个操作会减少锁的计数器值
        "local counter = redis.call('hincrby', KEYS[1], ARGV[3], -1);" +
		// 如果锁计数器值仍大于零,说明有其他线程持有锁,需要更新锁的过期时间
        "if (counter > 0) then " +
		// 更新锁的过期时间
        "redis.call('pexpire', KEYS[1], ARGV[2]); " +
		// 表示解锁操作成功但锁仍被其他线程持有
        "return 0; " +
        "else " +
		// 如果计数器值为零,说明当前线程是最后一个持有锁的线程
		// 删除锁
        "redis.call('del', KEYS[1]); " +
		// 发布解锁事件通知其他等待的线程
        "redis.call('publish', KEYS[2], ARGV[1]); " +
		// 表示解锁成功且锁已被完全释放
        "return 1; " +
        "end; " +
		// 如果条件不满足,返回 nil
        "return nil;",
        Arrays.asList(this.getRawName(), this.getChannelName()),
        new Object[]{LockPubSub.UNLOCK_MESSAGE, this.internalLockLeaseTime, this.getLockName(threadId)});
}

图解:

3. 其他锁

3.1 红锁和多锁的区别

RedLock 是一种更为复杂的分布式锁实现,保证了分布式环境中的高可用性和容错性,但需要多个 Redis 实例进行协调

多锁 的实现简单,但可靠性差,容易受到单点故障的影响,不适合对安全性和可靠性要求较高的应用

特点RedLock多锁(Multiple Locks)
实现方式使用多个独立的 Redis 实例,保证多数节点成功每个 Redis 实例独立设置锁
容错性高,支持在大多数节点上获取锁低,不能保证一致性和容错性
锁的获取需要在大多数实例中成功获取在任意一个实例上获取锁即可
安全性提供了更高的安全性和可靠性相对简单,但不适用于复杂场景
网络分区容忍性可以容忍部分节点失败,但不是所有不适合面对网络分区或节点故障的场景

3.2 简单演示

public static void main(String[] args) {
    String lockKey = "myLock";
    Config config = new Config();
    config.useSingleServer().setPassword("123456").setAddress("redis://127.0.0.1:6379");
    Config config2 = new Config();
    config.useSingleServer().setPassword("123456").setAddress("redis://127.0.0.1:6380");
    Config config3 = new Config();
    config.useSingleServer().setPassword("123456").setAddress("redis://127.0.0.1:6381");

    RLock lock = Redisson.create(config).getLock(lockKey);
    RLock lock2 = Redisson.create(config2).getLock(lockKey);
    RLock lock3 = Redisson.create(config3).getLock(lockKey);

    RedissonRedLock redLock = new RedissonRedLock(lock, lock2, lock3);

    try {
        redLock.lock();
    } finally {
        redLock.unlock();
    }
}

3.3 CAP之间的取舍

CAP 原则又称 CAP 定理, 指的是在一个分布式系统中, Consistency(一致性)、 Availability(可用性)、Partition tolerance(分区容错性), 三者不可得兼

一致性© : 在分布式系统中的所有数据备份, 在同一时刻是否同样的值(等同于所有节点访问同一份最新的数据副本)
可用性(A): 在集群中一部分节点故障后, 集群整体是否还能响应客户端的读写请求(对数据更新具备高可用性)
分区容忍性§: 以实际效果而言, 分区相当于对通信的时限要求. 系统如果不能在时限内达成数据一致性, 就意味着发生了分区的情况, 必须就当前操作在 C 和 A 之间做出选择

4. Redisson的限流功能

常见的限流功能:固定窗口算法、滑动窗口算法、漏桶算法、令牌桶算法

利用Redisson的令牌桶限流

@Test
void testLimiter() {
    // 创建一个限流器
    RRateLimiter rateLimiter = redissonClient.getRateLimiter("sanjin");
    // 初始化最大流速为每秒10个令牌
    rateLimiter.trySetRate(RateType.OVERALL, 10, 1, RateIntervalUnit.SECONDS);
    for (int i = 0; i < 20; i++) {
        // 尝试获取一个令牌
        boolean b = rateLimiter.tryAcquire();
        if (b) {
            System.out.println("成功获取第"+ i +"个令牌");
        } else {
            System.out.println("被第" + i + "次限流了");
        }
    }
}
---------------------------结果-----------------------------------
成功获取第0个令牌
成功获取第1个令牌
成功获取第2个令牌
成功获取第3个令牌
成功获取第4个令牌
成功获取第5个令牌
成功获取第6个令牌
成功获取第7个令牌
成功获取第8个令牌
成功获取第9个令牌
被第10次限流了
被第11次限流了
被第12次限流了
被第13次限流了
被第14次限流了
被第15次限流了
被第16次限流了
被第17次限流了
被第18次限流了
被第19次限流了

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/937171.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

火焰传感器与C++编程:精准检测火灾的技术实现

火灾是我们日常生活中一个不可忽视的安全隐患&#xff0c;而火灾报警系统的实现可以大大提高我们的安全保障。通过嵌入式技术和传感器&#xff0c;我们能够在第一时间识别火灾隐患并发出警报。火焰传感器作为一种专门用于火灾监测的传感器&#xff0c;能高效地通过红外线&#…

arcGIS使用笔记(无人机tif合并、导出、去除黑边、重采样)

无人机航拍建图之后&#xff0c;通过大疆智图软件可以对所飞行的区域的进行拼图&#xff0c;但是如果需要对拼好的图再次合并&#xff0c;则需要利用到arcGIS软件。下面介绍arcGIS软件在这个过程中常用的操作。 1.导入tif文件并显示的方法&#xff1a;点击“”图标进行导入操作…

VMware Workstation的有线连接消失了

进入/var/lib目录下 cd /var/lib 查看是否存在NetworkManager 文件 ls 将其删除&#xff0c;然后虚拟机reboot一下。 sudo rm -r NetworkManager reboot 解决了&#xff0c;可以联网

算法-动态数组-62.不同路径

一、题目 二、思路解析 1.思路&#xff1a; 对于找到目的地它的来源主要来源于目的地的上一格和目的地的左一格 2.常用方法&#xff1a; 无 3.核心逻辑&#xff1a; 1.处理边界&#xff1a; a.只向右移动&#xff0c;至始至终只有一条路径 for(int i0;i<m;i){dp[i][0]1; } …

C# 探险之旅:第三十节 - 类型class(继承Inheritance) —— 当“儿子”继承“老爸”的遗产

嘿&#xff0c;探险家们&#xff01;欢迎再次踏上我们的C#奇幻旅程。今天&#xff0c;我们要聊一个既有趣又实用的话题——继承&#xff08;Inheritance&#xff09;&#xff01;想象一下&#xff0c;如果你的“儿子”能够继承“老爸”的遗产&#xff0c;那编程世界里的对象们也…

Qt:Q_GLOBAL_STATIC实现单例(附带单例使用和内存管理)

转载 https://blog.csdn.net/m0_71489826/article/details/142288179 前言 本文主要写Q_GLOBAL_STATIC实现单例以及单例的释放&#xff0c;网上很多教程只有单例的创建&#xff0c;但是并没有告诉我们单例的内存管理&#xff0c;这就很头疼。 正文 使用 Qt 的 Q_GLOBAL_STA…

12.5【Java exp4】【DEBUG】

pro1 JwtAuthenticationFilter 类在两个不同的位置被定义了&#xff0c;导致Spring无法确定使用哪个定义。 为了解决这个问题&#xff0c;你可以采取以下几种方法之一&#xff1a; 检查类路径中的重复类&#xff1a; 确保 JwtAuthenticationFilter 类没有在多个地方被定义。检…

C++11语法解析(二)

可变参数模板 基本语法及原理 ・C11 支持可变参数模板&#xff0c;也就是说支持可变数量参数的函数模板和类模板&#xff0c;可变数目的参数被称为参数包&#xff0c;存在两种参数包&#xff1a;模板参数包&#xff0c;表示零或多个模板参数&#xff1b;函数参数包&#xff1…

《知识拓展 · 统一建模语言UML》

&#x1f4e2; 大家好&#xff0c;我是 【战神刘玉栋】&#xff0c;有10多年的研发经验&#xff0c;致力于前后端技术栈的知识沉淀和传播。 &#x1f497; &#x1f33b; CSDN入驻不久&#xff0c;希望大家多多支持&#xff0c;后续会继续提升文章质量&#xff0c;绝不滥竽充数…

轻量级日志管理平台:Grafana Loki搭建及应用(详细篇)

前言 Grafana Loki是Grafana Lab团队提供的一个水平可扩展、高可用性、多租户的日志聚合系统&#xff0c;与其他日志系统不同的是&#xff0c;Loki最初设计的理念是为了为日志建立标签索引&#xff0c;而非将原日志内容进行索引。 现在目前成熟的方案基本上都是&#xff1a;L…

【原生js案例】如何让你的网页实现图片的按需加载

按需加载&#xff0c;这个词应该都不陌生了。我用到你的时候&#xff0c;你才出现就可以了。对于一个很多图片的网站&#xff0c;按需加载图片是优化网站性能的一个关键点。减少无效的http请求&#xff0c;提升网站加载速度。 感兴趣的可以关注下我的系列课程【webApp之h5端实…

用于卫星影像间接RPC模型精化的通用光束法平差方法

引言 介绍了通用RPC模型的表达式&#xff0c;which has been down to death 描述了RPC模型产生误差的原因——主要与定义传感器方位的姿态角有关。 每个影像都会对应一个三维点云&#xff0c;但是对同一地物拍摄的不同影像对应出来的三维点云是不一样的&#xff0c;所以才需…

搭建Tomcat(一)---SocketServerSocket

目录 引入1 引入2--socket 流程 Socket&#xff08;应用程序之间的通讯保障&#xff09; 网卡(计算机之间的通讯保障) 端口 端口号 实例 client端 解析 server端 解析 相关方法 问题1&#xff1a;ServerSocket和Socket有什么关系&#xff1f; ServerSocket Soc…

玩转个性地图样式!蜂鸟视图蜂鸟云主题编辑器正式上线

当地图不再只是冷冰冰的数据呈现&#xff0c;而是具有美感、适应多场景需求的设计作品时&#xff0c;地图应用的价值也随之提升。 蜂鸟视图推出全新“主题编辑器”功能&#xff0c;助你轻松定制个性化地图样式&#xff0c;赋予地图更多创意与生命力&#xff01; 一、主题编辑器…

【Figma_01】Figma软件初始与使用

Figma初识与学习准备 背景介绍软件使用1.1 切换主题1.2 官方社区 设计界面2.1 创建一个项目2.2 修改文件名2.3 四种模式2.4 新增界面2.5 图层2.6 工具栏2.7 属性栏section透明度和圆角改变多边形的边数渐变效果描边设置阴影等特效拖拽相同的图形 背景介绍 Ul设计:User Interfa…

MATLAB中all,any函数的应用

all表示要查的范围内全非 0 0 0返回 1 1 1&#xff0c;否则返回 0 0 0 any表示要查的范围内有一个非 0 0 0返回 1 1 1&#xff0c;否则返回 0 0 0 向量和矩阵都可以使用&#xff0c;在矩阵中&#xff0c;可以通过1(看列)或2(看行)设置维度 a l l all all和 a n y any any函数…

.NET 9 已发布,您可以这样升级或更新

.NET 9 已经发布&#xff0c;您可能正在考虑更新您的 ASP.NET Core 应用程序。 我们将介绍更新应用程序所需的内容。从更新 Visual Studio 和下载 .NET SDK 到找出可能破坏应用程序的任何重大更改。 下载 .NET 9 SDK 这些是下载 .NET 9 SDK 所需的步骤。 更新 Visual Studi…

服务器数据恢复—热备盘上线过程中硬盘离线导致raid5阵列崩溃的数据恢复案例

服务器数据恢复环境&#xff1a; 两组分别由4块SAS接口硬盘组建的raid5阵列&#xff0c;两组raid5阵列划分LUN并由LVM管理&#xff0c;格式化为EXT3文件系统。 服务器故障&#xff1a; RAID5阵列中一块硬盘未知原因离线&#xff0c;热备盘自动激活上线替换离线硬盘。在热备盘上…

Mac上使用ln指令创建软链接、硬链接

在Mac、Linux和Unix系统中&#xff0c;软连接&#xff08;Symbolic Link&#xff09;和硬连接&#xff08;Hard Link&#xff09;是两种不同的文件链接方式。它们的主要区别如下&#xff1a; 区别&#xff1a; 硬连接&#xff1a; 不能跨文件系统。不能链接目录&#xff08;为…

PCIe学习笔记

PCIE高速串行数据总线 当拿到一块板子 比如你要用到PCIE 首先要看这块板子的原理图 一般原理图写的是 PCI express 表示PCIE 以下是Netfpga为例下的PCIE插口元件原理图 ![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/01dc604fbdc847e8998a978c83c7b2eb.png 一般主…