分布式锁理解

news2024/12/28 21:06:54

介绍分布式锁,我觉得从项目的背景入手把

在伙伴匹配系统中,我创建了一个定时任务,做为缓存预热的手段

这个具体原因在Redis-CSDN博客

接下来切入正题:

想象每个服务器都有一个定时任务,都要对数据库或者缓存进行操作

这会带来什么问题?

1:首先最先想到的肯定是资源浪费

2:其次如果我这个定时任务是一个插入操作,那是不是会导致数据库或者缓存有很多的重复数据

再或者是一个修改操作,那肯定会造成结果不唯一的错误

那知道了问题,我们就要去想怎么解决?

解决办法

其实这个问题有点像操作系统中的临界区的问题

如果学过操作系统应该就很容易想到:

讲到了锁,那这个锁是一般的锁能锁得住嘛?

锁:

Java 实现锁:synchronized 关键字,这个在学习线程的时候学习过,很容易理解,

不过有个问题,这个synchronized是只对一个JVM有效

我们这里的项目场景可以用这个方法解决嘛?

显然不行,因为你一个synchronized只能锁住一个服务器,有多个服务器同时操作,你还是没有办法

接下来就引出

分布式锁

但是分布式锁这个东西需要考虑的点有很多:

分布式锁的注意事项:

  1. 用完锁要释放(腾地方)
  2. 锁一定要加过期时间 
  3. 如果方法执行时间过长,锁提前过期了?
  4. 连锁效应:释放掉别人的锁
  5. 这样还是会存在多个方法同时执行的情况

解决方案:续期(看门狗机制)

分布式锁概述:

处理多个并发操作的情况,确保在分布式系统中的不同节点(不同服务器)上对共享资源的访问是有序的和安全的

什么意思呢:

有一个房间(数据库),有三台服务器ABC,他们都有这个定时任务,想要进去操作数据库

我们规定,ABC三个服务器需要抢夺一把锁,抢到的才能进入,进去之后并且还需要锁上门

这就和Java多线程很像。

分布式锁的实现:

说了这么多,我们应该怎么保证同一时间只有一个服务器能抢到锁呢?

核心思想:

先来的人先把数据改成自己的标识(服务器 ip),后来的人发现标识已存在,就抢锁失败,继续等待。

等先来的人执行方法结束,把标识清空,其他的人继续抢锁


下面介绍Redis实现分布式锁(我只会这个,等以后会得多了再来补充把)

分布式锁的实现:Redis

主要是基于命令:SETNX key value

塞滕克斯 |文档 --- SETNX | Docs (redis.io)

redis> SETNX mykey "Hello"
(integer) 1
redis> SETNX mykey "World"
(integer) 0
redis> GET mykey
"Hello"

用了setnx将mykey(key)设置为"Hello"(value)之后,不允许再更改了

更改会返回0。

当然如果我们直接用setnx命令去操作就很麻烦,就介绍下面这一个方法:

Redisson 实现分布式锁:

Redisson 是一个 java 操作 Redis 的客户端,提供了大量的分布式数据集来简化对 Redis 的操作和使用,可以让开发者像使用本地集合一样使用 Redis,完全感知不到 Redis 的存在。

还是贴一个官方文档:

https://github.com/redisson/redisson#quick-start

1:引入依赖:
<dependency>
   <groupId>org.redisson</groupId>
   <artifactId>redisson</artifactId>
   <version>3.32.0</version>
</dependency>  
2:配置:
package com.usercenter.usercenterproject.config;

/*
Redission的配置
 */

import lombok.Data;
import org.redisson.Redisson;
import org.redisson.api.RedissonClient;
import org.redisson.config.Config;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
@ConfigurationProperties
@Data
public class RedissionConfiguration {

    private String host;
    private String port;

    @Bean
    public RedissonClient redissonClient(){
        // 1. Create config object
        Config config = new Config();
        String address = String.format("redis://127.0.0.1:6379");
        config.useSingleServer().setAddress(address).setDatabase(3);
        // 2. Create Redisson instance
        // Sync and Async API
        RedissonClient redisson = Redisson.create(config);
        return redisson;
    }

}

这段直接复制官方文档,改一下地址就行。

还需要配置成单机模式config.useSingleServer()

3:Redisson得使用:       
a.获取锁对象getLock:
final RLock lock = redissonClient.getLock("shayu:user:recommend:lock");
b.尝试获取锁的操作tryLock:
lock.tryLock(0,-1,TimeUnit.MILLISECONDS)

tryLock 方法会立即返回 false,表示获取锁失败;

而当锁可用时,则尝试获取锁并返回 true,表示成功获取锁。

tryLock这个方法有三个参数:

  1. waitTime:等待时间,即尝试获取锁的最大等待时间。这个参数表示在尝试获取锁时最多愿意等待的时间长度,单位可以是毫秒或者其他时间单位。如果在等待时间内未能成功获取锁,则 tryLock 方法会返回 false。(这个得意思就是其它没有拿到这个锁得服务器,他们不能一直等待,等过了这个waitTime之后就会放弃抢锁)这里的waitTime可以设置为0,因为我们这个定时任务,只要有一个服务器获取了,其它服务器就不能再操作了

  2. leaseTime:租约时间,表示获取锁成功后的租约时长。这个参数指定了成功获取锁后的持有时间长度,即锁的有效期,单位也可以是毫秒或其他时间单位。当锁的持有时间达到租约时长后,系统会自动释放锁。(表示这个获得锁的服务器的最长拥有时间,过了这个拥有时间,这个服务器就必须释放锁)这个leaseTime这里设置为-1,这是因为后面的看门狗机制

  3. unit:时间单位,用于指定 waitTime 和 leaseTime 的时间单位,可以是 TimeUnit 中预定义的时间单位,例如 TimeUnit.MILLISECONDS 表示毫秒。这个参数用于确保 waitTime 和 leaseTime 的时间粒度符合需求。

c.释放锁unlock():

获取锁之后一定要释放

还有一个点,在释放锁之前要确认一下是否是自己的锁:lock.isHeldByCurrentThread()

        finally {
            if(lock.isHeldByCurrentThread()){
                System.out.println("unlock"+Thread.currentThread().getId());
                lock.unlock();
            }
        }

可以将释放锁这段代码放在finally,因为创建锁需要捕获异常,如果不在finally中释放,就可以会发生这个锁一直存在这种现象。

Redisson的看门狗机制:

我们设想一个场景,假设一台服务器获取了锁之后,要往数据库中插入数据,全部插入完的时间是30ms,不过锁的过期时间是20ms,那这样就会发生问题:

我操作还没做完,这个锁就过期了,这怎么行。

所以:Redisson的看门狗机制就可以解决这个问题:

这个机制可以自动延长锁的过期时间(只要你不释放锁,就会一直延长)

如何启动这个看门狗机制呢?

只要在tryLock的realseTime中传入-1,就可以启动Redisson的看门狗机制。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C语言作业5(学生管理系统C语言)

成学生管理系统 1> 使用菜单完成 2> 有学生的信息录入功能&#xff1a;输入学生个数&#xff0c;并将学生的姓名、分数录入 3> 查看学生信息&#xff1a;输出所有学生姓名以及对应的分数 4> 求出学习最好的学生信息&#xff1a;求最大值 5> 按姓名将所有学…

Hyper-V 性能监控工具

虚拟化是任何组织网络管理战略不可或缺的一部分&#xff0c;对于帮助提高网络效率和资源可用性至关重要。采用虚拟基础架构具有多种好处&#xff0c;例如最大限度地减少停机时间、降低运营成本和提高生产力。 在所有虚拟服务器中&#xff0c;Microsoft Hyper-V因其多功能性和可…

亚马逊测评如何实现不同账户拥有独立运行环境,提高成功率

测评之所以被认为是最快速有效的推广方式&#xff0c;是因为它能够迅速影响多个关键因素。通过测评&#xff0c;您能够快速提升关键词的转化率&#xff0c;从而获得更好的搜索排名。优质的评价有助于增加产品的权重和转化率&#xff0c;进一步提升排名。同时&#xff0c;增加的…

在Windows环境下安装pycharm

Python环境搭建 第一步下载安装python 等待安装完成 验证python是否安装成功 Python开发工具安装部署 JetBrains: Essential tools for software developers and teams PyCharm: the Python IDE for data science and web development 下载社区版本的PyCharm 双击打开下载好的…

花几千上万学习Java,真没必要!(一)

1、主流的操作系统&#xff1a; 目前主流的PC端操作系统包括Windows、Mac OS和Linux。其中Windows是由微软公司开发的操作系统&#xff0c;Mac OS是由苹果公司开发的操作系统&#xff0c;而Linux则是开放源代码的操作系统&#xff0c;它有很多的发行版&#xff1a;比如&#xf…

PostgreSQL(二十二)缓冲区管理器

目录 一、缓冲区概述 1、缓冲区结构 2、buffer_tag结构 3、Backend进程读取操作 4、写脏块 二、缓冲区管理器结构 1、第一层&#xff1a;Buffer Table layer&#xff08;缓冲区表层&#xff09; 2、第二层&#xff1a;Buffer Descriptor Layer&#xff08;缓冲区描述层…

Vue2-动画

1.transition过渡 | 用transition CSS做动画 Vue-transition文档&#xff1a;进入/离开 & 列表过渡 — Vue.js [用transition做CSS动画]Enter状态&#xff1a;JS Bin - Collaborative JavaScript Debugging Leave状态&#xff1a;JS Bin - Collaborative JavaScript Debug…

经验分享:征信查询多了会不会影响大数据综合评分?

很多人在申请贷款的时候&#xff0c;会有一个疑问&#xff0c;就是自己的征信没逾期&#xff0c;就是查询偏多一点&#xff0c;但能达到申贷要求&#xff0c;为什么还会被拒贷?其实就是大数据花了的原因&#xff0c;那征信查询多了会不会影响大数据综合评分呢?接下来本文就为…

【C++】继承最全解析(什么是继承?继承有什么用?)

目录 一、前言 二、什么是继承 ? &#x1f4a2;继承的概念&#x1f4a2; &#x1f4a2;继承的定义&#x1f4a2; &#x1f95d;定义格式 &#x1f347;继承权限 三、基类与派生类对象的赋值转换 四、继承的作用域 五、派生类中的默认成员函数 &#x1f4a2…

[leetcode]minimum-cost-to-reach-destination-in-time 规定时间内到达终点的最小费用

. - 力扣&#xff08;LeetCode&#xff09; class Solution { private:// 极大值static constexpr int INFTY INT_MAX / 2;public:int minCost(int maxTime, vector<vector<int>>& edges, vector<int>& passingFees) {int n passingFees.size();ve…

windows节点加入一个Linux集群组成的kubernetes集群

windows节点加入一个Linux集群组成的kubernetes集群 背景描述一.准备环境搭建二.Windows上docker安装三.配置Windows server2019方法一&#xff1a;自动配置安装方法二&#xff1a;手动配置安装失败处理过程 四.Windows的node加入Linux的kubernetes集群五.配置Linux的master 背…

在linux中查找 / 目录下的以.jar结尾的文件(find / -name *.jar)

文章目录 1、查找 / 目录下的以.jar结尾的文件 1、查找 / 目录下的以.jar结尾的文件 [rootiZuf6332h890vozldoxcprZ ~]# find / -name *.jar /etc/java/java-1.8.0-openjdk/java-1.8.0-openjdk-1.8.0.342.b07-1.el9_0.x86_64/lib/security/policy/limited/US_export_policy.ja…

初学SpringMVC之过滤器解决乱码

写个 login.jsp 页面 提交的 method 一般为 post&#xff08;写 get 不安全&#xff0c;地址栏上会显示&#xff09; action 表示提交后跳转的地址 &#xff08;不直接写控制器里的路径是因为配置 Tomcat 时设置了前缀路径&#xff0c;默认走 http://localhost:8080&#xf…

Linux ---gcc

c语言的链接类型&#xff1a; 动态链接&#xff1a; ll /lib64/libc-2.17.so 静态链接&#xff1a; ll /lib64/libc.a glibc-static安装-CSDN博客https://blog.csdn.net/itas109/article/details/104226783在Linux中下载c语言的静态库。 gcc test.c -o test.s -static 以…

代码随想录二刷7.22|977.有序数组的平方

暴力解法&#xff1a; ——如果想暴力解决这个问题的话&#xff0c;可以像题目那样&#xff0c;先将每一个元素平方&#xff0c;然后再排序 双指针&#xff1a; ——从题目中找到的信息&#xff1a;这是一个非递减顺序的整数数组&#xff0c;从例子中&#xff0c;可以容易看…

搭建NEMU与QEMU的DiffTest环境(Socket方式)

搭建NEMU与QEMU的DiffTest环境&#xff08;Socket方式&#xff09; 1 简述2 编译NEMU2.1 配置2.2 修改NEMU/scripts/build.mk2.3 修改isa_difftest_checkregs函数2.4 修改isa_pmp_check_permission函数2.5 编译 3 编译qemu-socket-difftest3.1 修改NEMU/scripts/isa.mk3.2 修改…

数据结构之单链表在不带标准头的情况下C,C#,C++分别怎么实现?

文章目录 单链表的概念单链表的操作单链表不带标准头结点示例C语言实现C#实现C实现 数据结构是计算机科学中非常重要的一部分&#xff0c;它帮助我们理解和操作数据。单链表是数据结构中的一种基本类型&#xff0c;它由一系列节点组成&#xff0c;每个节点包含数据域和指向列表…

强化学习编程实践-4-基于蒙特卡洛的方法

第3章给出了学习算法的基本思路&#xff1a;策略评估和策略改善。其中策略评估用到了以下的公式&#xff08;4.1&#xff09;&#xff1a; 策略改善则用了最简单的贪婪策略&#xff08;4.2&#xff09;&#xff1a; 为什么要用蒙特卡洛算法&#xff1f;先看公式4.1和4.2&#x…

图鸟UI框架在uni-app多端应用开发中的实践与应用

摘要&#xff1a; 随着移动互联网的蓬勃发展&#xff0c;跨平台应用开发已成为行业趋势。本文将探讨图鸟UI框架如何在uni-app开发环境下助力开发者高效构建多端应用&#xff0c;并通过具体案例展示其在实际项目中的应用效果。 一、引言 在移动应用开发领域&#xff0c;跨平台…

【错题集-编程题】栈和排序(栈 + 贪心)

牛客对于题目连接&#xff1a;栈和排序_牛客题霸_牛客网 (nowcoder.com) 一、分析题目 每次尽可能的先让当前需要的最大值弹出去。 二、代码 // 修改后的代码 class Solution { public:/*** 代码中的类名、方法名、参数名已经指定&#xff0c;请勿修改&#xff0c;直接返回方…