我的编程空间,编程开发者的网络收藏夹
学习永远不晚

从 Java 锁到分布式锁

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

从 Java 锁到分布式锁

本文主要是在学习 Java 锁以及分布式锁的源码后,做出的归纳总结。

1锁的最基本要素

为什么要使用锁?

关于为什么要使用锁这个问题,答案显而易见:“为了避免多线程并发冲突”。

在多线程中对公共数据的修改,必须要保证只有线程在进行操作。这里的公共数据可以是公共变量,也可以是数据库中的一行数据。

锁的基本要素

知道为什么要加锁之后,就可以得出加锁的基本要素:

  • 锁标志:怎么样才算加锁成功?
  • 锁持有者:是哪个线程加的锁?
  • 锁重入:自己加了锁之后,再次加锁?
  • 锁并发:并发加锁失败的线程该怎么办?
  • 锁释放:用完锁,该怎么释放?

简单来说应该就是这些要素,遗漏之处,欢迎补充。

2加锁标志

加锁标志,就是需要一个标志来表示是否加锁成功,并且这个加锁标志要保证原子性。

  • synchronized 底层是 C++ 实现的,在 ObjectMonitor 对象中有一个 _count 参数用来标识是否持有锁。
  • ReentrantLock 基于 AQS 实现,其中 state 表示线程是否持有锁。ReentrantReadWriteLock 同样基于 AQS 实现,其中 state 高 16 位表示读锁,低 16 位表示写锁。
  • Redisson 是基于 Redis 的 Hash 数据结构实现的,其中加锁 key 是否存在,可以表示是否加锁。
  • Curator 基于 ZooKeeper 临时顺序节点实现,判断加锁路径下是否存在该节点,来表示是否加锁。

3锁持有者

  • 锁持有者,肯定是当前线程,但是在分布式锁中还需要加上机器,用来防止服务之间的线程冲突。
  • synchronized 在 ObjectMonitor 对象中 _owner 是指获得锁的线程。
  • ReentrantLock 和 ReentrantReadWriteLock 是在 AQS 的 Node 节点中有 Thread 对象,用来表示获得锁的线程。
  • Redisson 在 Hash 数据结构的 field 字段存放的是 UUID:ThreadId,从而保证在多个服务实例时防止并发冲突。
  • Curator 创建的临时顺序节点包含 UUID,表示加锁机器,结构比如 /locks/lock_01/_c_UUID-lock-0000000000。

4锁重入

当获得锁的线程再次尝试获取锁的时候,保证需要计数。

  • synchronized 会对 _count 进行累加,CAS 更新。
  • ReentrantLock 会对 AQS 的 state 进行累加,CAS 更新。
  • Redisson 使用 Lua 脚本,对 Hash 结构的 value 进行累加。
  • Curator 是在 Java 代码中维护了一个 AtomicInteger 类型的 lockCount 字段,用来表示重入。

5锁等待

  • synchronized 并发加锁失败线程会自旋等待,涉及到偏向锁、轻量级锁、重量级锁的锁升级流程。
  1. 刚开始是无锁的
  2. 偏向锁:一段同步代码一直被一个线程访问,这个线程自动获取锁,大多数都是由同一个线程获取锁,这就会出现偏向锁。目的是只有一个线程执行同步代码块时提高性能,JDK 6 后在 JVM 中默认开启。对象头标志位(01-无锁) 是否偏向锁标志(1-是偏向锁)
  3. 轻量级锁:锁是偏向锁时,被其他线程访问,偏向锁就升级为轻量级锁,其他线程通过自旋形式尝试获取锁 对象头标志位 00
  4. 重量级锁:只有一个线程等待,该线程是在自旋等待获取锁。当自旋一定次数或者一个持有锁一个自旋时来了第三个线程,就会升级为重量级锁。对象头标志位 10
  • ReentrantLock 会放到 AQS 双向同步队列中,监听上一个节点是否为虚拟头结点,是则尝试获取锁。
    • 非公平锁新线程会默认参加抢锁,公平锁会先查看队列是否为空。
    • 自旋等待时会使用 LockSupport.park() 方法,这时候会让出 CPU 资源,其他线程会调用 LockSupport.unpark()。
    • 可以使用 tryLock 方法设置时间,在指定时间内获取锁失败或者被中断,则会返回加锁失败。
  • Redisson 并发加锁,失败线程会获取到当前锁的超时时间,然后通过 Semaphore tryAcquire 方法阻塞一定时间后,再次尝试获取锁。
    • 公平锁会额外使用 Redis 的 List 数据结构来当做线程等待队列,使用 sorted set 有序集合数据结构来存放等待线程的顺序(score 是超时时间戳)。
  • Curator 并发加锁会直接创建临时顺序节点,然后监听顺序节点中自己的上一个节点(防止羊群效应),默认是公平锁。

6锁释放

  • synchronized 不需要手动释放。
  • ReentrantLock 对 state 递减为 0 后,唤醒后续节点,有后续节点需要调用 LockSupport.unpark(s.thread)。
  • Redisson 主动释放直接删除 key 即可。超时释放即服务宕机,没有看门狗续租了,指定时间后,Redis Key 就会被自动释放。
  • Curator 主动释放会删除节点,如果服务宕机了,节点会被自动删除。

7总结

本文从多个角度总结分析了锁和分布式锁的基本要素,同样基于 MySQL 等数据库的锁可以参考实现。

 本文转载自微信公众号「程序员小航」,可以通过以下二维码关注。转载本文请联系程序员小航公众号。

 

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

从 Java 锁到分布式锁

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

从 Java 锁到分布式锁

在并发编程中常用到 synchronized 以及 ReentrantLock 锁,在业务开发过程中也可能会用到分布式锁,分布式锁常用框架的就是基于 Redis 实现的分布式锁框架 Redisson 和 基于 Zookeeper 实现的分布

【分布式】分布式锁

目录 一、分布式锁介绍二、基于 Redis 实现分布式锁1. 如何基于 Redis 实现一个最简易的分布式锁?2.为什么要给锁设置一个过期时间?3. 如何实现锁的优雅续期?4. 如何实现可重入锁? 一、分布式锁介绍 单机多线
2023-08-17

ZooKeeper 分布式锁 Curator 源码 05:分布式读写锁和联锁

Curator 同样支持分布式读写锁[1] 和联锁[2],只需要使用 InterProcessReadWriteLock 即可,来一起看看它的源码以及实现方式。

Java 从零实现属于你的 Redis 分布式锁

我们想要解决分布式系统中的并发问题,就需要引入分布式锁的概念。
JavaRedis开发2024-12-03

Redisson分布式锁

文章目录 一、Redisson简单介绍二、Redisson简单使用1. maven引用2. RedisConfig配置3. StockRedissonService4. 测试 三、Redisson源码1. 加锁2. 解锁3. 自
2023-08-19

redis分布式锁与zk分布式锁的对比分析

目录Redis实现分布式锁原理能实现的锁类型注意事项 zk实现分布式锁原理能实现的锁类型两种锁的对比在分布式环境下,传统的jvm级别的锁会失效,那么分布式锁就是非常有必要的一个技术,一般我们可以通过redis,zk等技术来实现我们的分布式锁
2022-11-18

DBA_2PC_PENDING中的分布式锁-解锁

运行shell脚本后,会生成 roll.sql文件。
2023-06-06

Java分布式锁如何实现

这篇“Java分布式锁如何实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Java分布式锁如何实现”文章吧。一、分布式锁介
2023-07-05

Redisson 分布式锁源码之公平锁加锁

默认的加锁逻辑是非公平的。在加锁失败时,线程会进入 while 循环,一直尝试获得锁,这时候是多线程进行竞争。就是说谁抢到就是谁的。

Java基于Redis实现分布式锁

分布式锁可以基于很多种方式实现,比如zookeeper、redis...。不管哪种方式,他的基本原理是不变的:用一个状态值表示锁,对锁的占用和释放通过状态值来标识。一、为什么Redis可以方便地实现分布式锁1、Redis为单进程单线程模式,采用队列模式将并发访
Java基于Redis实现分布式锁
2015-09-14

Java如何实现ZooKeeper分布式锁

这篇文章主要介绍了Java如何实现ZooKeeper分布式锁,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。什么是分布式锁1、在我们进行单机应用开发,涉及并发同步的时候,我们往
2023-06-29

分布式锁的原理及Redis怎么实现分布式锁

这篇文章主要介绍“分布式锁的原理及Redis怎么实现分布式锁”,在日常操作中,相信很多人在分布式锁的原理及Redis怎么实现分布式锁问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”分布式锁的原理及Redis怎么
2023-02-02

从Curator实现分布式锁的源码再到羊群效应

Curator是一款由Java编写的,操作Zookeeper的客户端工具,在其内部封装了分布式锁、选举等高级功能。

Redisson 分布式锁源码 10:读写锁

Redisson 还支持可重入读写锁,允许在分布式场景下,同时有多个读锁和一个写锁处于加锁状态。

Redis分布式锁之红锁的实现

目录一、问题二、办法三、原理四、实战一、问题分布式锁,当我们请求一个分布式锁的时候,成功了,但是这时候slave还没有复制我们的锁,masterDown了,我们的应用继续请求锁的时候,会从继任了master的原slave上申请,也会成功。
2022-08-09

Redisson分布式锁之加解锁详解

目录引言锁的可重入性加锁锁续命释放锁引言2023的金三银四来的没想象中那么激烈,一个朋友前段时间投了几十家,多数石沉大海,好不容易等来面试机会,就恰好被问道项目中关于分布式锁的应用,后涉及Redisson实现分布式锁的原理,答不上来。锁的
2023-03-19

Redisson分布式锁之加解锁源码分析

这篇文章主要介绍“Redisson分布式锁之加解锁源码分析”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Redisson分布式锁之加解锁源码分析”文章能帮助大家解决问题。锁的可重入性我们都知道,Ja
2023-07-05

Redisson 分布式锁源码之公平锁排队加锁

Redis sorted set 有序集合数据结构:存放等待线程的顺序,分数 score 用来是等待线程的超时时间戳。

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录