Redisson 分布式锁 - RLock、RReadWriteLock、RSemaphore、RCountDownLatch(配置、使用、原理)

目录

前言

Redisson 分布式锁

环境配置

1)版本说明

2)依赖如下

3)配置文件如下

4)项目配置

RLock

1)使用方式

2)加锁解释

3)加锁时手动设置时间

4)加锁时,到底要不要手动设置过期时间?(最佳实践)

RReadWriteLock

1)使用方式

2)加锁原理

RSemaphore

1)使用方式

2)信号原理

RCountDownLatch

1)使用方式

2)原理解释


前言


前面讲过一篇 Redisson 分布式锁的底层原理,而这篇文章着重实战,因此对原理不清楚的,可以看看我之前的文章:http://t.csdnimg.cn/LU5ED

 

Redisson 分布式锁


环境配置

1)版本说明

  • SpringBoot:3.2.5
  • Redisson:3.25.0

 

2)依赖如下

<!--        <dependency>-->
<!--            <groupId>org.springframework.boot</groupId>-->
<!--            <artifactId>spring-boot-starter-data-redis</artifactId>-->
<!--        </dependency>-->

        <!--redisson 依赖整合了 StringRedisTemplate,因此 data redis 依赖就可以删除了-->
        <dependency>
            <groupId>org.redisson</groupId>
            <artifactId>redisson-spring-boot-starter</artifactId>
            <version>3.25.0</version>
        </dependency>

 

3)配置文件如下

spring:  
  data:
    redis:
      host: env-base
      port: 6379

Ps:实际上还有一种配置方式就是自己配置 RedissonClient 的 Bean,注入给容器

 

4)项目配置

分布式锁带来的效果演示,会通过 jmeter 来进行测试.  服务这边会先通过 网关,在负载均衡到集群的实例上.  因此这里我们来配置一下集群的每个实例信息.

这里为了方便观察,准备了两个实例:

RLock

1)使用方式

如果使用的 Redisson 的 Boot Starter 依赖的话,只需要在 yml 按照本文配置,然后在需要的地方注入 RedissionClient 即可使用(非 Boot Starter 依赖需要自己配置 RedissonClient).

@RestController 
@RequestMapping("/product/lock")
class Test(
    private val redisson: RedissonClient
) {

    @GetMapping("/test1")
    fun test1(): String {
        //1.获取一把锁,只要名字一样,就是同一把锁
        val lock: RLock = redisson.getLock("my-lock")

        //2.加锁
        lock.lock()

        try {
            println("加锁成功,执行业务..." + Thread.currentThread().id)
            Thread.sleep(10000) //模拟耗时任务
        } catch (e: Exception) {
            e.printStackTrace()
        } finally {
            //3.解锁
            lock.unlock()
            println("解锁成功!" + Thread.currentThread().id)
        }

        return  "ok!"
    }

}

 Ps:除此之外,还有 redisson 的 ReentrantLock 中还提供了 tryLock() 方法有以下两种重载方式

  • boolean tryLock():尝试加锁,如果当前锁被占用,则直接放弃并返回 false.
  • boolean tryLock(long time, TimeUnit unit):如果当前锁被占用,则会等待,知道到达我们设置的过期时间 time 还没拿到锁,就放弃并返回 false.

 

2)加锁解释

a)RLock 就类似于 JUC 中的 ReentrantLock,是一个可重入锁(同一个线程对同一个资源连续加锁两次不会死锁).

b)加锁实际上就是在 redis 上添加了一个 key-value,并且默认加锁的过期时间为 30s,如下图:

c)如果业务处理实践比默认加锁时间长怎么办?这里会有一个看门狗机制,只要拿到锁,就会开启一个定时任务,每隔 10s 就会自动续约.  因此不用担心业务时间长的问题.  

d)如果代码还没有执行到解锁,程序就挂了,会不会死锁?不会的,锁是有默认的过期时间,即使没有执行到解锁逻辑,锁也会自动删除.(这里我自己测试了一下,貌似新版的 Redisson 中会检测程序是否挂了,如果挂了,就会把这个锁立即删除掉)

3)加锁时手动设置时间

a)使用如下:

b)注意:

如果我们手动指定了过期时间,那么即使业务没有执行完,也不会自动续约.  也就是说,无论如何,到期自动解锁.

4)加锁时,到底要不要手动设置过期时间?(最佳实践)

a)最佳实践:

使用 lock.lock(30, TimeUnit.SECONDS) 手动设置 30s 过期时间.  

b)原因:

如果我们手动设置过期时间,就省掉了续约的操作(有一定的开销).

再者,真的会有某一个业务逻辑需要执行 30s 的时间么?如果真的有,这个程序大概率是出问题了.  到了 30s 后解锁,反而还避免了 “死等” 问题.

RReadWriteLock

1)使用方式

a)写操作(写锁)

    @GetMapping("/write")
    fun write(): String {
        val rwLock: RReadWriteLock = redisson.getReadWriteLock("rw-lock")
        var result = ""
        //1.获取写锁
        val wLock = rwLock.writeLock()
        //2.写操作用写锁,读操作用读锁
        wLock.lock()
        try {
            Thread.sleep(10000)
            result = UUID.randomUUID().toString()
            redisTemplate.opsForValue().set("uuid", result)
        } catch (e: Exception) {
            e.printStackTrace()
        } finally {
            wLock.unlock()
        }
        return "ok! uuid: $result"
    }

b)读操作(读锁)

    @GetMapping("/read")
    fun read(): String {
        val rwLock: RReadWriteLock = redisson.getReadWriteLock("rw-lock")
        var result: String? = ""
        //1.获取读锁
        val rLock = rwLock.writeLock()
        //2.写操作用写锁,读操作用读锁
        rLock.lock()
        try {
            result = redisTemplate.opsForValue().get("uuid")
        } catch (e: Exception) {
            e.printStackTrace()
        } finally {
            rLock.unlock()
        }
        return "ok! uuid: $result"
    }

Ps:读锁,写锁 这里也额外提供了 tryLock() 方法,来尝试加锁(原理上面讲过)

2)加锁原理

读写锁保证了读操作和读操作之间不会加锁,而读操作和其他任何操作都会加锁.   使得在读多写少的业务场景中,效率大大提升.

Redission 提供的 ReadWriteLock 也是这个原理:

  • 读 + 读: 读操作和读操作之间不会出现脏数据问题,因此相当于无锁,只会在 redis 中记录当前读锁,他们都会同时加锁成功.
  • 写 + 读:如果先写,此时紧接着又进行读操作,可能出现脏数据的问题,因此会阻塞等待写锁释放.
  • 写 + 写:写操作和写操作之间可能出现脏数据问题,因此也是阻塞等待.
  • 读 + 写:由于你读的时候,另一个线程又来写,也会出现脏数据的问题,因此也必须要阻塞等待读锁释放.

RSemaphore

1)使用方式

    @GetMapping("/park")
    fun park(): String {
        //这里的 RSemaphore 就相当于是一个停车场,刚开始的没有车位(初始信号量为 0)
        val park: RSemaphore = redisson.getSemaphore("park")
        //1.获取一个信号,相当于占了一个停车位,车位 - 1
        park.acquire()
        //2.执行业务
        //...
        return "park ok!"
    }

    @GetMapping("/go")
    fun go(): String {
        val park = redisson.getSemaphore("park")
        //1.释放一个信号,相当于让出了一个车位,车位 + 1
        park.release()
        return "go ok!"
    }

Ps:这里也有一个额外的方法 tryAcquire(),尝试申请资源 

2)信号原理

a)redisson.getSemaphore("park") 这里实际上就是在 redis 上添加一个 key-value,key 就是我们自定义的 "park" 字符串,value 就是信号量,初始情况下为 0. 

b)情况分析:

情况一:刚开始的时候如果 线程A 进行 acquire(),由于信号量为 0,只能阻塞等待.  接着如果有 线程B 进行 release(),就会释放一个信号量,也就是信号量 + 1,此时 线程A 发现有一个信号来了,他就直接消费掉了

情况二:刚开始的时候如果 线程A 进行 release(),此时信号量 + 1,总共信号量为 1,接着如果有线程B 来进行 acquire() ,就会直接消费掉这个信号,此时信号量 - 1,总共信号量为 0.

Ps:信号量也可以做分布式限流

RCountDownLatch

1)使用方式

例如有 5 个选手比赛,要求所有选手到达终点之后才可以宣布比赛结束.

    @GetMapping("/match")
    fun match(): String {
        val match = redisson.getCountDownLatch("match")
        //1.设置计数器的初始值为 5 (想象成有 5 名选手赛跑)
        match.trySetCount(5)
        //2.等待所有资源全被消费 (等待 5 名选手全部跑完)
        match.await()
        return "比赛结束!"
    }

    @GetMapping("/gogogo/{id}")
    fun gogogo(@PathVariable("id") id: Long): String {
        val match = redisson.getCountDownLatch("match")
        //1.计数器 - 1 (一名选手到达终点)
        match.countDown()
        return "选手 $id 号到达终点!"
    }

2)原理解释

a)redisson.getCountDownLatch("match") 这里实际上就是给 redis 存了一个 key-value,key 就是我们自定义的 "match" 字符串,value 就是计数器.

b)match.trySetCount(5) 就是给这个计数器设置了一个初始值为 5.

c)match.await() 会一直阻塞住,直到计数器的值减为 0.

d)match.countDown() 让计数器 - 1.

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

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

相关文章

JVM运行数据区-Java堆

Java堆 堆区&#xff08;Heap区&#xff09;是JVM运行时数据区占用内存最大的一块区域&#xff0c;每一个JVM进程只存在一个堆区&#xff0c;它在JVM启动时被创建&#xff0c;JVM规范中规定堆区可以是物理上不连续的内存&#xff0c;但必须是逻辑上连续的内存。 1、堆区是线程…

王学岗鸿蒙开发(北向)——————(一)鸿蒙开发环境的搭建与ArkTs介绍

1&#xff0c;鸿蒙系统开始研发的时间是在2012年。 2&#xff0c;目前鸿蒙有两个开发:HarmonyOS和OpenHarmony,前者内聚AOSP(Android的东西)&#xff0c;前者是双框架结构&#xff0c;后者不是双框架结构&#xff0c;没有内置安卓。 3&#xff0c;Harmony地址 4&#xff0c;我们…

训练Pytorch深度学习模型出现StopIteration

训练一个深度学习检测模型&#xff0c;突然出现&#xff1a; 是因为next(batch_iterator)&#xff0c;可能迭代器读出来的数据为空。 # load train data# 原先代码images, targets next(batch_iterator)# 更改为&#xff1a;try:images, targets next(batch_iterator)except…

对接钉钉登陆步骤

背景 之前事情较少的时候&#xff0c;帮公司写过一个系统&#xff0c; 这个系统的话主管有要求要对接钉钉登陆。 话不多说我们直接开干。流程 先进入开发者平台点击开发者后台 没有组织的 我们先在手机上先创建一个组织 创建完成后&#xff0c;就可以看到这个组织了 创建…

Michael.W基于Foundry精读Openzeppelin第56期——VestingWallet.sol

Michael.W基于Foundry精读Openzeppelin第56期——VestingWallet.sol 0. 版本0.1 VestingWallet.sol 1. 目标合约2. 代码精读2.1 constructor()2.2 beneficiary() && start() && duration() && receive() payable2.3 released() && releasable(…

加密经济浪潮:探索Web3对金融体系的颠覆

随着区块链技术的快速发展&#xff0c;加密经济正在成为全球金融领域的一股新的浪潮。而Web3作为下一代互联网的代表&#xff0c;以其去中心化、可编程的特性&#xff0c;正深刻影响着传统金融体系的格局和运作方式。本文将深入探讨加密经济对金融体系的颠覆&#xff0c;探索We…

普通人下班可以做点什么补偿家用

你我&#xff0c;或者说大多数的都是普通人&#xff0c;每个人都在为了生活奔波&#xff0c;没有惊天动地的才华&#xff0c;也没有一夜暴富的运气&#xff0c;但我们依然可以通过自己的双手和智慧&#xff0c;为家庭添上一份温馨。白天的工作往往只能满足基本的生活需求&#…

IIS7整合Tomcat9服务器,并搭建ASP+PHP+JSP完整运行环境

本文以Windows Vista系统为例&#xff0c;详细讲解IIS7整合Tomcat服务器&#xff0c;同时支持ASPPHPJSP三种Web动态网页技术的方法。 Vista系统自带的IIS版本为7.0&#xff0c;能安装的IE浏览器的最高版本为IE9。IE9也是Vue2前端框架支持的最低浏览器版本。 【准备工作】 去微…

第六讲:AD、DA的工作原理及实现、运放电路

DA 数模转换器 (DAC) 数模转换器&#xff08;Digital-to-Analog Converter&#xff0c;简称DAC&#xff09;是一种将数字信号转换为模拟信号的电子装置。DAC在各种电子设备中广泛应用&#xff0c;如音频设备、通信系统、测量设备和控制系统中。以下是DAC的主要概念和应用。…

已发【镜像仿真篇】ESXi镜像仿真教程

【镜像仿真篇】ESXi镜像仿真教程 我以为不会再有使用FTK Imager低版本的时候&#xff0c;毕竟Arsenal Image Mounte是我目前遇到的最强镜像挂载软件&#xff0c;直到这次遇到了这个ESXi镜像仿真的时候一直报错—【蘇小沐】 1、实验环境 FTK Imanger &#xff0c;[v3.1.1.8]V…

李廉洋:6.4-6.5黄金原油再次走低,美盘行情分析及最新策略。

黄金消息面分析&#xff1a;全球债券周二上涨&#xff0c;呼应美债隔夜的涨势。美联储或早降息的押注增强了主权债务的吸引力。澳大利亚和新西兰10年期债券收益率下跌至少8个基点&#xff0c;先前数据显示&#xff0c;美国5月份工厂活动萎缩的速度加快。日本10年期债券收益率下…

01_深度学习基础知识

1. 感知机 感知机通常情况下指单层的人工神经网络,其结构与 MP 模型类似(按照生物神经元的结构和工作原理造出来的一个抽象和简化了模型,也称为神经网络的一个处理单元) 假设由一个 n 维的单层感知机,则: x 1 x_1 x1​ 至 x n x_n xn​ 为 n 维输入向量的各个分量w 1 j…

云原生架构案例分析_4.某电商业务云原生改造

名称解释&#xff1a; AHAS&#xff1a;应用高可用服务&#xff08;Application High Availability Service&#xff09;是一款专注于提高应用高可用能力的SaaS产品&#xff0c;主要包含多活容灾、故障演练和流量防护三个独立的功能模块。其中流量防护已迁移至微服务治理服务MS…

windows hash简介

一、hash简介 1、Windows系统使用两种方法对用户的密码进行哈希处理。它们分别是LAN Manager(LM)哈希和 NT LAN Manager(NTLM)哈希 2、所谓哈希(hash)&#xff0c;就是使用一种加密函数进行计算后的结果。这个加密函数对一个任意长度的 字符串数据进行一次数学加密函数运算…

计网ppt标黄知识点整理第(4)章节——谢希仁版本、期末复习自用

路由器&#xff1a;查找转发表&#xff0c;转发分组。 IP网的意义&#xff1a;当互联网上的主机进行通信时&#xff0c;就好像在一个网络上通信一样&#xff0c;看不见互连的各具体的网络异构细节。如果在这种覆盖全球的 IP 网的上层使用 TCP 协议&#xff0c;那么就…

Spring运维之boot项目开发关键之日志操作以及用文件记录日志

日志基础 日志 在企业级开发中还是比较重要的 我们来写一个日志 RestController RequestMapping("/books") public class Controller {//创建记录日志的对象private static final Logger log LoggerFactory.getLogger(Controller.class);GetMappingpublic String …

开源基于Rust编写的Web服务器

基于 RUST 的 WEB 资源服务器 Github 地址 LTPP-GIT 地址 官方文档 该项目于 2024 年 5 月 1 日开始开发 预期功能 功能支持情况当前情况多线程支持是是服务支持配置化是是防盗链支持是是gzip 支持是是反向代理支持是是自定义状态码对应资源文件是是日志支持是是负载均衡支…

【Python绘画】画笑脸简笔画

本文收录于 《一起学Python趣味编程》专栏&#xff0c;从零基础开始&#xff0c;分享一些Python编程知识&#xff0c;欢迎关注&#xff0c;谢谢&#xff01; 文章目录 一、前言二、代码示例三、知识点梳理四、总结 一、前言 本文介绍如何使用Python的海龟画图工具turtle&#…

BUUCTF [XMAN2018排位赛]通行证 1

BUUCTF:https://buuoj.cn/challenges 题目描述&#xff1a; 得到的 flag 请包上 flag{} 提交。来源&#xff1a;https://github.com/hebtuerror404/CTF_competition_warehouse_2018 密文&#xff1a; a2FuYmJyZ2doamx7emJfX19ffXZ0bGFsbg解题思路&#xff1a; 1、两个等号…

CentOS-内网搭建FTP-Server

一、镜像选择 1、 Centos-everting或者DVD 2、7.5 7.6 7.9 均可 二、安装步骤 1、其余步骤和普通安装一致。 2、最重要的一步为“软件选择” 1、勾选FTP、文件以及存储服务器、性能以及开发工具。 三、FTPServer搭建 1、关闭防火墙 systemctl stop firewalld or 通过21和20…