Redis04-分布式锁

news2024/11/18 15:44:34

目录

Redis实现分布式锁

分布式锁的工作流程

Redis实现分布式锁

Redission的watch dog

Redis分布式锁的合理应用


Redis实现分布式锁

在单节点的服务器中,java中的synchronized机制是处于JVM层面的,只能保证线程之间的同步。而实际的服务部署中,我们通常会使用多节点的集群部署方式,而要保证多个节点中的不同进程对数据的同步操作,就需要使用分布式锁。

使用分布式锁的主要目的是在分布式环境中确保多个节点之间的协同操作,避免由于并发访问而导致的数据不一致、竞态条件等问题。以下是使用分布式锁的一些常见原因:

  1. 避免并发问题: 在分布式系统中,多个节点可能同时对共享资源进行访问。使用分布式锁可以确保对关键资源的访问在同一时刻只有一个节点拥有,避免并发问题,如数据不一致、丢失更新等。

  2. 保护共享资源: 分布式系统中的共享资源,例如数据库、缓存、文件等,可能同时被多个节点访问。通过使用分布式锁,可以保护这些共享资源,确保在任何时刻只有一个节点能够修改它们。

  3. 防止死锁: 在分布式系统中,由于网络分区、节点故障等原因,可能发生死锁。通过使用分布式锁,可以避免因为多节点间的协同操作导致的死锁问题。

  4. 控制访问顺序: 在某些场景下,需要对共享资源的访问进行有序控制,确保一些操作的执行顺序。分布式锁可以用于实现对资源的顺序访问。

  5. 实现分布式协调: 分布式锁是实现分布式协调和同步的一种重要机制。例如,在分布式任务调度、分布式事务等场景中,分布式锁可以用于协调不同节点的操作。

  6. 防止重复执行: 在一些需要幂等性操作的场景下,分布式锁可以用于防止同一操作被重复执行。

  7. 避免“多写”问题: 当多个节点同时写入同一数据时,可能导致“多写”问题,即最后写入的值覆盖了之前的写入。分布式锁可以用于保护写入操作,确保只有一个节点能够写入。

分布式锁的工作流程

Redis实现分布式锁

Redis实现分布式锁主要利用的是Redis的setnx命令。setnc是Set if no exists的简写。

CenOs7:0>setnx lock 1 # 首次设置lock值1
"1"  # 设置成功返回1
CenOs7:0>setnx lock 1 # lock已存在,给lock赋值
"0"  # 设置失败返回0,不做任何操作
CenOs7:0>get lock # 获取lock的值
"1"  # lock值为1
CenOs7:0>del lock # 删除lock,释放锁
"1"  # 删除成功,返回1
CenOs7:0>get lock # 获取lock的值
null # lock为空
CenOs7:0>setnx lock 1 # 设置lock值为1
"1"  # 设置成功

需要注意的是,setnx的方式是不支持重入锁的。而且锁的持有时间很难控制。单纯地依赖expire设置过期时间是不现实的。

Redission的watch dog

Redission引入了“看门狗”的机制,目的是处理锁的续期问题。

代码示例:

import org.redisson.Redisson;
import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.redisson.config.Config;
​
import java.util.concurrent.TimeUnit;
​
public class RedissonWatchdogExample {
​
    public static void main(String[] args) throws InterruptedException {
        // 创建 Redisson 客户端
        Config config = new Config();
        config.useSingleServer().setAddress("redis://127.0.0.1:6379");
        RedissonClient redisson = Redisson.create(config);
​
        // 获取分布式锁
        RLock lock = redisson.getLock("myLock");
​
        try {
            // 尝试获取锁并设置锁的过期时间为10秒
            // boolean isLocked = lock.tryLock(10, 30, TimeUnit.SECONDS);
            // 如果设置了第二个参数[锁的持续时间]就默认不使用watchdog
            // redission中的lock是依赖了底层lua脚本的原子性实现的
            boolean isLocked = lock.tryLock(10, TimeUnit.SECONDS);
​
            if (isLocked) {
                // 获取锁成功,执行业务逻辑
                System.out.println("Lock acquired. Executing business logic...");
​
                // 模拟业务逻辑执行时间
                Thread.sleep(5000);
​
                // 业务逻辑执行完毕,不需要手动释放锁,Watchdog会自动续期或者释放锁
​
            } else {
                System.out.println("Failed to acquire the lock within the specified time");
            }
​
        } finally {
            // 关闭 Redisson 客户端
            redisson.shutdown();
        }
    }
}

创建了一个名为"myLock"的分布式锁,并使用tryLock方法尝试获取锁,设置锁的过期时间为10秒。在获取锁成功后,执行了一段模拟的业务逻辑,并且无需手动释放锁,因为Watchdog会在锁过期时自动续期或释放锁。

需要注意的是,Redisson Watchdog是默认开启的,你不需要额外的配置。在正式的生产环境中,建议你根据实际需求进行更详细的配置,比如设置Watchdog的检查间隔、锁的失效时间等参数。

Redis分布式锁的合理应用

Redis分布式锁是在多个节点之间实现互斥访问共享资源的一种机制。它可以应用于各种场景,以下是一些合理的应用场景:

  1. 分布式任务调度: 当有多个节点需要执行定时任务或者周期性任务时,可以使用分布式锁确保同一时刻只有一个节点在执行任务,避免重复执行或者竞态条件。

  2. 分布式缓存更新: 当多个节点需要同时更新某个缓存或者共享的数据时,可以使用分布式锁确保只有一个节点能够进行更新操作,防止并发问题。

  3. 分布式事务: 在分布式事务场景下,可能需要对一些关键资源进行加锁,以确保在事务执行期间,其他事务不会修改这些资源,从而保证事务的隔离性。

  4. 防止重复操作: 在一些场景下,可能需要防止某个操作被重复执行。通过使用分布式锁,可以确保同一时刻只有一个节点能够执行这个操作,防止重复操作。

  5. 分布式限流: 在某些情况下,需要对请求进行限流,确保系统不被过度请求。通过使用分布式锁,可以控制某个时间窗口内只有一个节点能够处理请求,实现简单的限流效果。

  6. 分布式协同: 当多个节点需要协同工作时,可以使用分布式锁确保各个节点之间的操作按照规定的顺序进行,防止不一致或混乱的情况发生。

  7. 分布式任务队列: 在分布式环境中,多个节点可能需要消费同一个任务队列。通过使用分布式锁,可以确保只有一个节点能够成功获取并消费队列中的任务,避免重复消费。

  8. 资源竞争: 当多个节点需要竞争某个资源时,可以使用分布式锁来进行竞争,确保只有一个节点能够成功占用资源,如分布式锁用于实现分布式秒杀场景。

需要注意的是,在使用分布式锁时,要注意锁的超时策略、锁的释放等问题,以确保系统的稳定性和性能。此外,选择成熟的分布式锁实现库,如Redisson、Curator等,可以减少潜在的问题,提高开发效率。

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

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

相关文章

Spring面试题:(六)Spring注解开发原理

ioc过程 发现只要将bean注册到BeanDefinitionMap中就可以创建bean对象 如何将xml配置的bean注册到BeanDefinitionMap 通过注解注册的bean过程一样 注册bean的接口:BeanDefinitionRegistryPostProcessor 开启组件扫描的两种方式:xml和注解 xml方式…

Unity之NetCode多人网络游戏联机对战教程(8)--玩家位置同步

文章目录 前言添加相机玩家添加对应组件服务端权威(server authoritative)客户端权威(client authoritative)服务端同步位置阅读与理解PlayerTransformSync.csNetworkVariableUploadTransformSyncTransform 后话 前言 承接上篇&a…

vColorPicker与vue3-colorPicker——基于 Vue 的颜色选择器插件

文章目录 前言样例特点 一、使用步骤?1. 安装2.引入3.在项目中使用 vcolorpicker 二、选项三、事件四、问题反馈问题所在安装引入例子效果图 前言 vColorPicker——官网 vColorPicker——GitHub 样例 vColorPicker是基于 Vue 的一款颜色选择器插件,仿照…

主题模型LDA教程:主题数选取 困惑度perplexing

文章目录 LDA主题数困惑度1.概率分布的困惑度2.概率模型的困惑度3.每个分词的困惑度 LDA主题数 LDA作为一种无监督学习方法,类似于k-means聚类算法,需要给定超参数主题数K,但如何评价主题数的优劣并无定论,一般采取人为干预、主题…

电子工程师的焊接技法总结

基础学习视频如下: 1 老司机焊接纯干货分享,让你焊接不迷路,很适合零基础小白_哔哩哔哩_bilibili 焊接常用工具 1 焊锡丝 按照粗细来分的话,有粗焊锡,有细焊锡,细焊锡一般适合比较精细的焊接。 按照是否含铅…

吃透 Spring 系列—Web部分

目录 ◆ Spring整合web环境 - Javaweb三大组件及环境特点 - Spring整合web环境的思路及实现 - Spring的web开发组件spring-web ◆ web层MVC框架思想与设计思路 ◆ Spring整合web环境 - Javaweb三大组件及环境特点 在Java语言范畴内,web层框架都是基于J…

win环境Jenkins部署前端项目

今天分享win环境Jenkins部署前端vue项目,使用的版本jenkins版本Jenkins 2.406版本。 前提是jenkins安装好了,通用配置已经配置好了,可以参考上两篇博客。 1、前端项目依赖nodejs,需要安装相关插件 点击进入 安装成功标准 jenki…

Home Assistant使用ios主题更换背景

Home Assistant使用ios主题、更换背景 lovelace-ios-dark-mode-theme 默认前置情况,1、已安转HACS插件2、搜索安装 IOS Dark Mode Theme1)第一、二步应该很容易实现,configuration.yaml文件很容易被找到2)而本人在进行第三步操作时…

在vue3中使用Element-plus的图标

首先安装Element-Plus-icon # 选择一个你喜欢的包管理器# NPM $ npm install element-plus/icons-vue # Yarn $ yarn add element-plus/icons-vue # pnpm $ pnpm install element-plus/icons-vue 如何使用 Element-Plus-icon官方文档链接Icon 图标 | Element Plus (element-…

Zyxel NBG2105 身份验证绕过

直接访问如下payload则会以管理员身份跳转到 home.htm页面 ​​/login_ok.htm漏洞证明 查看本页面的cookie,login为1 文笔生疏,措辞浅薄,望各位大佬不吝赐教,万分感谢。 免责声明:由于传播或利用此文所提供的信息、…

antlr4踩坑记录

一. syntax error: ‘<’ came as a complete surprise to me while matching alternative 参考这个issue&#xff0c;antlr版本必须得是4.6 下载链接&#xff1a;http://www.antlr.org/download/antlr-4.6-complete.jar 二.org.antlr.v4.analysis.LeftRecursiveRuleTrans…

如何在ModelScope社区魔搭下载所需的模型

本篇文章介绍如何在ModelScope社区下载所需的模型。 若您需要在ModelScope平台上有感兴趣的模型并希望能下载至本地&#xff0c;则ModelScope提供了多种下载模型的方式。 使用Library下载模型 若该模型已集成至ModelScope的Library中&#xff0c;则您只需要几行代码即可加载…

【Vue3】scoped 和样式穿透

我们使用很多 vue 的组件库&#xff08;element-plus、vant&#xff09;&#xff0c;在修改样式的时候需要进行其他操作才能成功更改样式&#xff0c;此时就用到了样式穿透。 而不能正常更改样式的原因就是 scoped 标记。 scoped 的渲染规则&#xff1a; <template>&l…

Ubuntu查看Python某个包的具体路径

使用命令&#xff1a; python(版本号) -m pip show (包)这里的Location就是这个包所在的路径。同时它还列出了这个包的版本的信息。

吴恩达《机器学习》8-3->8-4:模型表示I、模型表示II

8.3、模型表示I 一、大脑神经网络的基本原理 为了构建神经网络模型&#xff0c;首先需要理解大脑中的神经网络是如何运作的。每个神经元都可以被看作是一个处理单元或神经核&#xff0c;它包含多个输入&#xff08;树突&#xff09;和一个输出&#xff08;轴突&#xff09;。…

【见缝插针】射击类游戏-微信小程序项目开发流程详解

还记得小时候玩过的见缝插针游戏吗&#xff0c;比一比看谁插得针比较多&#xff0c;可有趣了&#xff0c;当然了&#xff0c;通过它可以训练自己的手速反应&#xff0c;以及射击水平&#xff0c;把握时机&#xff0c;得分越高就越有成就感&#xff0c;相信小朋友们会喜欢它的&a…

pointnetgpd复现

参考&#xff1a; Installation Instructions — Dex-Net 0.2.0 documentation Install git clone https://github.com/lianghongzhuo/PointNetGPD.git 添加环境变量 gedit ~/.bashrc #添加下面这一行 export PointNetGPD_FOLDER$HOME/code/PointNetGPD #然后source source…

k8s 1.28.3 使用containerd

文章目录 环境说明最终结果环境配置时钟同步 主机名称配置主机名解析关闭swap安装ipvs 安装containerd安装containerd生成配置修改配置开启containerd服务 安装runc安装k8s安装kubelet kubeadm kubectl获取kubernetes 1.28组件容器镜像 拉取镜像初始化集群方法一&#xff08;不…

【4】Gradle-快速入门使用【Gradle多模块项目详解】

目录 【4】Gradle-快速入门使用【Gradle多模块项目详解】创建多项目构建添加子项目命名建议 项目依赖项项目路径不同模块的build.gradle配置 子项目之间共享构建逻辑公约插件跨项目配置buildSrc开发公约插件 调整多模块项目配置修改项目树的元素 了解Gralde配置时间和执行时间并…

API 集成测试工具Hitchhiker 0.1.1 正式发布

Hitchhiker 是一款开源的 Restful Api 集成测试工具&#xff0c;你可以在轻松部署到本地&#xff0c;和你的 team 成员一起管理 Api。 能做什么 * Team 协作开发 Api * Api 历史修改记录及支持 diff 展示 * 支持多环境变量及运行时变量 * 支持 Schedule 及批量 run * 不同…