实现分布式锁

背景

分布式锁是一种用于协调分布式系统中多个节点之间并发访问共享资源的机制。在分布式系统中,由于存在多个节点同时访问共享资源的可能性,需要使用分布式锁来保证数据的一致性和正确性。

今天要实现的是分布式场景中的互斥类型的锁。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

下面时分布式锁的实现重点
在这里插入图片描述

借助MySQL实现

主要利用MySQL唯一键和唯一性约束来实现互斥性

表结构

DROP TABLE IF EXISTS `dislock`;CREATE TABLE `dislock` (`id` int(11) unsigned NOT NULL AUTO_INCREMENT	COMMENT '主键',`lock_type` varchar(64) NOT NULL COMMENT '锁类型',`owner_id` varchar(255) NOT NULL COMMENT '持锁对象',`update_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,PRIMARY KEY (`id`),UNIQUE KEY `idx_lock_type` (`lock_type`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 COMMENT='分布式锁表';

创建一个名为dislock的表,用于存储分布式锁的信息。该表包含以下字段:

  • id:主键,自增长的整数类型。
  • lock_type:锁类型,字符串类型,用于标识锁的类型。
  • owner_id:持锁对象,字符串类型,用于标识持有锁的客户端或其他相关信息。
  • update_time:更新时间,时间戳类型,表示最后一次更新该行数据的时间。

在上面的SQL语句中,idx_lock_type是一个唯一索引,它定义在lock_type字段上,确保表中不会有两个相同的lock_type值。这样可以保证同一类型的锁在表中只有一条记录,避免了同一客户端重复获取锁的情况。

加锁和解锁

INSERT INTO dislock (`lock_type`, `owner_id`) VALUES ('act_lock', 'ad2daf3');
DELETE FROM dislock WHERE `lock_type` = 'act_lock' AND `owner_id` = 'ad2daf3'; // 要确保谁加锁谁解锁

如果要用MySQL实现的话,**还需要开一个进程(要求计算型高可用),来管理超时,**不断去查询updatetime,如果该锁超过了最长允许存在的时间,则需要删除数据库对应的该行。

如果要实现可重入的话,那么考虑再增加一个数据库字段来实现。

借助Redis实现

在Redis中,可以使用SET命令来实现分布式锁。具体实现方法如下:

  1. 使用SET命令尝试在Redis中设置一个键,例如lock:resource,并将其值设置为一个唯一的标识符,例如当前进程的ID或一个随机字符串。可以使用NX选项来确保只有在该键不存在时才进行设置,从而实现加锁操作。

  2. 如果SET命令返回了OK,则说明该键被成功设置,当前进程获得了锁;否则,说明该键已经被其他进程持有,当前进程没有获得锁,需要等待一段时间后重试。

  3. 在完成操作后,使用DEL命令删除该键,释放锁。

需要注意的是,在使用SET命令设置键的值时,应该设置一个过期时间,以防止锁被持有进程意外终止而无法释放。可以使用EX选项来设置过期时间,例如SET lock:resource <identifier> EX 10表示将锁的过期时间设置为10秒。

另外,为了避免锁的误删除,应该使用与设置键的标识符不同的值来删除锁。可以使用Lua脚本来实现原子性的加锁和解锁操作,从而避免并发问题。例如,下面是一个简单的Lua脚本,实现了基本的加锁和解锁操作:

-- 加锁操作
if redis.call('SET', KEYS[1], ARGV[1], 'EX', ARGV[2], 'NX') thenreturn 1
elsereturn 0
end-- 解锁操作
if redis.call('GET', KEYS[1]) == ARGV[1] thenreturn redis.call('DEL', KEYS[1])
elsereturn 0
end

我们今天主要考虑下面的方法
当使用 Redis 的 Redlock 算法时,需要考虑以下关键步骤和注意事项:

  1. 选择 N 个 Redis 实例

    • 选择的 Redis 实例应该分布在不同的物理服务器或者逻辑节点上,以确保它们不会受到单点故障的影响。
    • 这些实例可以是独立的 Redis 节点,也可以是 Redis 集群中的节点。
  2. 获取当前时间

    • 在进行锁操作之前,首先需要获取一个精确的当前时间戳。这可以通过系统时间或者某种网络时间协议(如NTP)来获得。
  3. 尝试在每个实例上获取锁

    • 对于每个选定的 Redis 实例,使用相同的锁键和唯一的随机值(例如 UUID)执行 SET 操作,并设置适当的超时时间和标识符,以确保只有持有锁的客户端才能释放锁。
    • 设置锁的超时时间是为了避免锁被永久性地持有,即使持有锁的客户端出现故障。
  4. 计算获取锁所需的时间

    • 计算获取锁所需的时间,主要包括获取锁的时间、网络延迟以及时钟漂移等。这些因素都会影响锁的有效性。
  5. 判断是否成功获取锁

    • 如果大多数实例(至少超过半数)成功获取了锁,并且用于获取锁的时间没有超过超时时间,则认为获取锁成功。
    • 如果无法获得大多数实例的支持,可能需要重试或者放弃锁操作。
  6. 释放锁

    • 在完成任务后,需要在所有实例上执行删除操作来释放锁。

注意事项

  • 网络分区:在面对网络分区时,需要考虑如何处理部分 Redis 实例无法通信的情况。这可能需要引入额外的逻辑来检测并处理这种情况。
  • 时钟漂移:由于不同的服务器上的时钟可能存在不同程度的漂移,需要谨慎处理时钟同步问题,以确保锁的超时时间计算准确。
  • Redis 实例故障:需要考虑 Redis 实例的故障处理,包括如何发现故障实例并采取适当的措施。

总之,实现 Redlock 算法需要综合考虑各种因素,并根据具体情况进行调整,以确保系统在分布式环境下能够正确、高效地运行。

代码实现参考:
链接: https://github.com/jacket-code/redlock-cpp

其他的方法未完待续,后续补充

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

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

相关文章

Tensorflow 入门基础——向LLM靠近一小步

进入tensflow的系统学习&#xff0c;向LLM靠拢。 目录 1. tensflow的数据类型1.1 数值类型1.2 字符串类型1.3 布尔类型的数据 2. 数值精度3. 类型转换3.1 待优化的张量 4 创建张量4.1 从数组、列表对象创建4.2 创建全0或者1张量4.3 创建自定义数值张量 5. 创建已知分布的张量&…

luceda ipkiss教程 56:画多端口螺旋线

案例分享&#xff1a;画多端口螺旋线 注&#xff1a;spiral的长度不是真实长度&#xff0c;具体可以参考教程28 代码如下&#xff1a; from si_fab import all as pdk import ipkiss3.all as i3 import numpy as np from scipy.constants import piclass SpiralCircular(i3.P…

linux perf工具使用

参考文章Linux性能调优之perf使用方法_perf交叉编译-CSDN博客 perf是一款Linux性能分析工具。比如打流性能优化的时候&#xff0c;就能够看到是哪些函数消耗的cpu高 那么linux如何编译perf工具呢&#xff1f; perf工具编译 进入perf目录下linux-3.16/tools/perf make ARCH…

HarmonyOS 应用开发入门

HarmonyOS 应用开发入门 前言 DevEco Studio Release版本为&#xff1a;DevEco Studio 3.1.1。 Compile SDK Release版本为&#xff1a;3.1.0&#xff08;API 9&#xff09;。 构建方式为 HVigor&#xff0c;而非 Gradle。 最新版本已不再支持 &#xff08;”Java、JavaScrip…

React16源码: React中的updateHostRoot的源码实现

HostRoot 的更新 1 &#xff09;概述 HostRoot 是一个比较特殊的节点, 因为在一个react应用当中它只会有一个 HostRoot, 它对应的 Fiber 对象是我们的 RootFiber 对象重点在于它的更新过程 2 &#xff09;源码 定位到 packages/react-reconciler/src/ReactFiberBeginWork.js…

【Qt】信号和槽

需要云服务器等云产品来学习Linux的同学可以移步/-->腾讯云<--/-->阿里云<--/-->华为云<--/官网&#xff0c;轻量型云服务器低至112元/年&#xff0c;新用户首次下单享超低折扣。 目录 一、Qt中的信号和槽 1、信号 2、槽 3、Q_OBJECT 二、Qt中的connect函…

Windows下 VS2022 编译OpenSSL 库

SSL是Secure Sockets Layer(安全套接层协议)的缩写,可以在Internet上提供秘密性传输。Netscape公司在推出第一个Web浏览器的同时,提出了SSL协议标准。其目标是保证两个应用间通信的保密性和可靠性,可在服务器端和用户端同时实现支持。已经成为Internet上保密通讯的工业标准…

k3s x GitLab Runner Operator,GitLab CI 云原生构建新体验

GitLab CI 是非常常用的一款 CI/CD 工具&#xff0c;只需要在 .gitlab-ci.yml 文件中用 YAML 语法编写 CI/CD 流水线即可。而 GitLab CI 能够运行的关键组件是 GitLab Runner。GitLab Runner 是一个轻量级、高扩展的代理&#xff0c;主要用来执行 GitLab CI/CD 流水线中的 Job&…

stm32 FOC系列 直流有刷控制原理

1、直流有刷驱动板 使用三极管搭建的简易 H 桥电路&#xff0c;如图 5.3.1 所示&#xff1a; 图 5.3.1 是使用三极管搭建的简易 H 桥电路&#xff0c;其中 MOTOR 表示直流有刷电机&#xff0c; Q1、 Q2、 Q3 和 Q4 为 4 个三极管&#xff0c;其中 Q1 和 Q3 接在了电源正极&…

[AutoSar]BSW_OS 08 Autosar OS_内存保护

一、 目录 一、关键词平台说明一、内存保护的概念 关键词 嵌入式、C语言、autosar、OS、BSW 平台说明 项目ValueOSautosar OSautosar厂商vector &#xff0c;芯片厂商TI 英飞凌编程语言C&#xff0c;C编译器HighTec (GCC) >>>>>回到总目录<<<<&l…

matlab appdesigner系列-常用13-标签

标签&#xff0c;用来显示各类文本 此示例&#xff0c;就是在标签之外的画布上单击鼠标左键&#xff0c;显示王勃的《滕王阁诗》 操作如下&#xff1a; 1&#xff09;将2个标签拖拽到画布上&#xff0c;并修改相应文字。将第二个标签的右侧文本信息中的Wordwrap打开&#xf…

【C++】文件操作

文件操作 一、文本文件&#xff08;一&#xff09;写文件读文件 二、二进制文件&#xff08;一&#xff09;写文件&#xff08;二&#xff09;读文件 程序运行时产生的数据都属于临时数据&#xff0c;程序一旦运行结束都会被释放&#xff0c;通过文件可以将数据持久化&#xff…

记录一下uniapp 集成腾讯im特别卡(未解决)

uniapp的项目运行在微信小程序 , 安卓 , ios手机三端 , 之前这个项目集成过im,不过版本太老了,0.x的版本, 现在需要添加客服功能,所以就升级了 由于是二开 , 也为了方便 , 沿用之前的webview嵌套腾讯IM的方案 , 选用uniapp集成ui ,升级之后所有安卓用户反馈点击进去特别卡,几…

基于 OpenVINO, yolov5 推理

OpenVINO 是英特尔开发的一款功能强大的深度学习工具包&#xff0c;可实现跨多个硬件平台的优化神经网络推理。在本文中&#xff0c;我们讨论了 OpenVINO 的特性和优势&#xff0c;以及它如何与领先的计算机视觉平台 Viso Suite 集成&#xff0c;以构建和交付可扩展的应用程序。…

Mysql数据库cpu飙升怎么解决

排查过程 &#xff08;1&#xff09;使用top命令观察&#xff0c;确定是mysql导致还是其他原因。 &#xff08;2&#xff09;如果是mysql导致的&#xff0c;show processlist&#xff0c;查看session情况&#xff0c;确定是不是有消耗资源的sql在运行。 &#xff08;3&#xf…

【OSG案例详细分析与讲解】之十四:【立方体贴图】

文章目录 一、【立方体贴图】前言二、【立方体贴图】实现效果三、【立方体贴图】创建立方体贴图1、实现目标2、实现步骤3、核心代码4、知识要点5、TextureCubeMap详讲6、osg::TexGen详讲7、TexEnvCombine详讲四、【立方体贴图】程序1、程序代码2、qt pro文件五、【立方体贴图】…

李沐深度学习-激活函数/多层感知机文档

multilayer perceptron (MLP)&#xff1a;多层感知机(多层神经网络) (hidden layer)隐藏层&#xff1a; 介于输入层和输出层之间的网络层 输入层不涉及计算&#xff0c;如果一个神将网络由三层组成&#xff0c;则多层感知机层数为2 多层感知机中隐藏层和输出层都是全连接 隐藏…

如何快速打开github

作为一个资深码农&#xff0c;怎么能不熟悉全球最大的同性交友社区——github呢&#xff0c;但头疼的是github有时能打开&#xff0c;有时打不开&#xff0c;这是怎么回事&#xff1f; 其实问题出在github.com解析DNS上&#xff0c;并不是需要FQ。下面提供一个方法&#xff0c;…

hot100:07接雨水

题目链接&#xff1a; 力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台 算法思想&#xff1a; 这里采取的是暴力解法和双指针的解法&#xff0c;但是这个题目还有其他的两种解法&#xff08;单调栈和动态规划&#xff0c;同学可以自行了解&#xff…

Golang 中高级工程师学习笔记

闭包的作用 闭包&#xff08;Closure&#xff09;是一种函数值&#xff0c;它可以引用在其外部定义的变量。闭包允许这些变量保持在函数内部&#xff0c;而不是被每次调用时重新创建。闭包的作用主要体现在以下几个方面 封装&#xff1a; 闭包允许函数访问其外部作用域中的变…