分布式锁理解

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

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

这个具体原因在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;@Beanpublic RedissonClient redissonClient(){// 1. Create config objectConfig 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 APIRedissonClient 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.mzph.cn/pingmian/45528.shtml

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

相关文章

Coze触发器:触发任务的Python接口源码

文章目录 📖 介绍 📖🏡 演示环境 🏡📒 Coze触发器 📒📝 触发器接口源码⚓️ 相关链接 ⚓️📖 介绍 📖 自动化,一个在现代软件开发中不可或缺的概念,它让我们的生活和工作变得更加高效。Coze也支持定时任务/触发任务,通过触发器,我们可以更自由的控制Bot去…

【cocos creator】2.x,伪3d拖拽,45度视角,60度视角,房屋装扮

伪3d拖拽&#xff0c;45度视角,60度视角 工程下载&#xff1a;&#xff08;待审核&#xff09; https://download.csdn.net/download/K86338236/89530812 dragItem2.t s import mapCreat2 from "./mapCreat2";const { ccclass, property } cc._decorator; /*** 拖拽…

【web]-信息收集-空白页面

打开是一张图 查看源码&#xff0c;发现就一个链接是有用信息&#xff0c;用目录扫描工具&#xff0c;没有发现有价值的信息。 F12&#xff0c;查看请求和相应信息&#xff0c;在响应头中发现了信息。 还有一个小技巧&#xff1a;点击手机图标&#xff0c;可以切换到手机模式中…

Spring相关的面试题

1、spring中bean的生命周期 spring bean的生命周期主要分为三大类 &#xff0c;分别是创建-》使用-〉销毁。 在三大类下面又可以分为5个小类。分别是 实列化-〉初始化-》组册destruction回调-〉使用-〉销毁 这这其中 初始化也可以细分为 设置属性值&#xff0c;前置处理&#…

UDP协议介绍和作用

什么是UDP? UDP是User Datagram Protocol的简称&#xff0c;中文名是用户数据报协议&#xff0c;是OSI参考模型中的传输层协议&#xff0c;它是一种无连接的传输层协议&#xff0c;提供面向事务的简单不可靠信息传送服务。 UDP的正式规范是IETF RFC768。UDP在IP报文的协议号是…

手撸俄罗斯方块(四)——渲染与交互

手撸俄罗斯方块&#xff08;四&#xff09;——渲染与交互 如何渲染游戏界面 我们知道&#xff0c;当我们看到页面先呈现图像时&#xff0c;实际上看到的是一张图片&#xff0c;多张图片按照一定的刷新频率进行切换&#xff0c;则变成了动态的视频。当刷新频率超过24Hz时&…

DWG文件发布至IIS后无法下载和预览解决办法

问题描述 DWG文件发布至IIS后无法下载和预览 原因分析&#xff1a; iis里面需要添加扩展 解决方案&#xff1a; 在服务器端IS属性的HTTP头下的MIME内容中添加扩展名“.dwg” MIME类型填入application/acad

29.PLL(锁相环)-IP核的调用

&#xff08;1&#xff09;PLL IP核的简介&#xff1a; Phase Locked Loop&#xff0c;即锁相环&#xff0c;是最常用的IP核之一&#xff0c;其性能强大&#xff0c;可以对输入到FPGA的时钟信号进行任意分频、倍频、相位调整、占空比调整&#xff0c;从而输出一个期望时钟。锁相…

JVM学习(day1)

JVM 运行时数据区 线程共享&#xff1a;方法区、堆 线程独享&#xff08;与个体“同生共死”&#xff09;&#xff1a;虚拟机栈、本地方法栈、程序计数器 程序计数器 作用&#xff1a;记录下次要执行的代码行的行号 特点&#xff1a;为一个没有OOM&#xff08;内存溢出&a…

C语言:指针详解(4)

作者本人由于大一下学期事情繁多&#xff0c;大部分时间都在备赛&#xff0c;没有时间进行博客撰写&#xff0c;如今已经到了暑假时间&#xff0c;作者将抓紧每一天的时间进行编程语言的学习&#xff0c;由于目前作者已经进行到了C的学习&#xff0c;C语言阶段的学习与初阶数据…

Tensorflow入门实战 T08-Vgg16网络进行猫狗识别

目录 1、前言 2、代码 3、运行结果 4、反思 本文为&#x1f517;365天深度学习训练营 中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 | 接辅导、项目定制 1、前言 本周学习内容为&#xff0c;采用自己设置的vgg-16网络进行猫狗识别&#xff0c;并非官网提供的…

【公益案例展】亚运天穹——践行亚运理念,筑牢安全防线

‍ 安恒信息公益案例 本项目案例由安恒信息投递并参与数据猿与上海大数据联盟联合推出的 #榜样的力量# 《2024中国数据智能产业最具社会责任感企业》榜单/奖项”评选。 大数据产业创新服务媒体 ——聚焦数据 改变商业 杭州第19届亚运会是中国第三次举办亚洲最高规格的国际综合…

217.贪心算法:加油站(力扣)

代码解决 class Solution { public:int canCompleteCircuit(vector<int>& gas, vector<int>& cost) {int curtotol 0; // 当前累积油量int tatol 0; // 总的油量减去总的花费油量int start 0; // 起始加油站的索引// 遍历所有加油站for (int i 0; i &…

5款文案生成神器,自动一键生成原创文案

文案在我们的生活中随处可见&#xff0c;好的文案内容不仅可以为企业带来销售转化&#xff0c;而且还能提升品牌的影响力&#xff0c;因此文案的重要性可想而知&#xff0c;对于文案创作者来说&#xff0c;写作好的文案不是轻松容易的事&#xff0c;但如果把这个任务交给文案生…

Python中的null是什么?

在知乎上遇到一个问题&#xff0c;说&#xff1a;计算机中的「null」怎么读&#xff1f; null正确的发音是/n^l/&#xff0c;有点类似四声‘纳儿’&#xff0c;在计算机中null是一种类型&#xff0c;代表空字符&#xff0c;没有与任何一个值绑定并且存储空间也没有存储值。 P…

MySQL CONCAT函数的简单使用

CONCAT函数用于将mysql中查询多列的值拼成一列显示&#xff0c; 使用示例&#xff1a; SELECT CONCAT(attr_name,"&#xff1a;",attr_value) FROM pms_sku_sale_attr_value WHERE sku_id1; 上面SQL语句使用CONCAT函数将attr_name、attr_value两列的值拼成一列&am…

动态sql 单选变多选

实体类 添加数组存储值 private ArrayList tssjfjList; <!-- <if test"tssjfj ! null and tssjfj ! ">and tssjfj #{tssjfj}</if>--><if test"tssjfjList ! null and tssjfjList.size() > 0">AND tssjfj IN<fo…

JeecgBoot 前端 vue3 项目,配置项目多页面入口

前端 vue3配置项目多页面入口 1.项目根目录新建home.html <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport" content"widthdevice-width, initial-scale1.0" /><…

Rust Hello

首先还是安装&#xff1a; 一定要换源&#xff0c;否则真的太慢了。 curl --proto https --tlsv1.2 -sSf https://sh.rustup.rs | sh 就是~/.cargo/config [source.crates-io] # 替换成你偏好的镜像源 replace-with tuna# 清华大学 5mb [source.tuna] registry "htt…

数字信号处理教程(4)—— 离散傅里叶变换DFT

今天继续数字信号处理的学习&#xff0c;本次我们将来聊聊离散傅里叶变换DFT(Discrete Fourier Transform)。DFT是数字信号处理领域中应 用最为广泛的离散变换。DFT将一个序列x(n)映射到频率域。DFT 的许多性质都与对模拟信号进行傅里叶变换的性质相同。里面包含着大量的算法值…