Java 标准库 API 系列之 StampedLock

本贴最后更新于 691 天前,其中的信息可能已经东海扬尘

在学习蚂蚁金服实现的 Raft 算法实现 sofa-jraft 项目中第一次接触到这个锁。jraft 使用 StampedLock 来实现对于 raft group 的路由信息的线程安全。

StampedLock 是 Java 8 中引入的新的锁机制,它支持三种模式的访问控制,分别是读模式、写模式和乐观读模式,它可以在高并发的情况下提供更好的性能和吞吐量。

在 StampedLock 中,每个线程都需要获取一个戳(stamp)来进行读或写操作,这个戳是一个 64 位的数字,其中高 16 位代表一个版本号,低 48 位代表一个序号。线程可以使用 tryOptimisticRead()方法来获取一个乐观读锁,这种方式不会阻塞线程,只是获取一个戳,如果获取到的戳发生了变化,就意味着有其他线程已经修改了共享资源,这时候需要重新获取锁。

StampedLock 还支持读锁和写锁。读锁是共享锁,多个线程可以同时持有读锁,并发地读取共享资源,而写锁是独占锁,只有一个线程可以持有写锁,其他线程需要等待写锁释放才能继续执行。

StampedLock 的优势在于,对于读多写少的场景,使用乐观读锁可以避免大量线程阻塞,提高系统的吞吐量;对于写操作比较频繁的场景,使用 StampedLock 可以减少锁的竞争,提高系统的性能。

API

StampedLock 有三种锁的获取方式:

  1. 读锁(readLock()):多个线程可以同时持有读锁,只要没有线程持有写锁。读锁是乐观的且不可重入的。通过调用 tryOptimisticRead()方法获取读锁,如果锁可用,会返回一个非零的标记(stamp)。
  2. 写锁(writeLock()):只有一个线程可以持有写锁,它排除所有的读锁和写锁。写锁是悲观的且可重入的。通过调用 writeLock()方法获取写锁。
  3. 乐观读锁(Optimistic Read Lock):乐观读锁是一种特殊的读锁,它允许其他线程在获取读锁之前修改共享数据,但在验证阶段检测到数据被修改时,需要重新获取锁。乐观读锁的获取方式是调用 tryOptimisticRead()方法,如果锁可用,会返回一个非零的标记(stamp)。

下面是一个简单的 StampedLock 的使用示例:

import java.util.concurrent.locks.StampedLock; public class Point { private double x; private double y; private final StampedLock lock = new StampedLock(); public void setPoint(double x, double y) { long stamp = lock.writeLock(); try { this.x = x; this.y = y; } finally { lock.unlockWrite(stamp); } } public double distanceFromOrigin() { long stamp = lock.tryOptimisticRead(); double currentX = x; double currentY = y; if (!lock.validate(stamp)) { stamp = lock.readLock(); try { currentX = x; currentY = y; } finally { lock.unlockRead(stamp); } } return Math.sqrt(currentX * currentX + currentY * currentY); } }

在上述示例中,setPoint() 方法使用写锁来更新坐标。写锁是 排他锁,当一个线程获取写锁时,其他线程无法获取读锁或写锁。

distanceFromOrigin() 方法使用乐观读锁尝试获取点的当前坐标,并计算点到原点的距离。如果乐观读锁验证失败(即期间有写操作),则通过获取悲观读锁来重新读取坐标。

这个例子展示了 StampedLock 的乐观读锁和悲观写锁的使用方式,通过合理地选择锁的模式,可以提供更高的并发性能,并且允许读操作和写操作同时进行,而不会相互阻塞。

校验是否有写操作 是如何判断的

StampedLock 的写锁有一个版本号,读锁没有版本号。因此,StampedLock 可以通过判断当前读锁的版本号是否等于写锁的版本号来检查是否存在写锁。如果版本号相同,说明当前存在写锁,反之,则没有写锁。需要注意的是,在使用 StampedLock 时,如果读锁和写锁是在不同的线程中获取的,那么在判断是否存在写锁时可能存在一定的延迟,因为写锁的版本号可能已经变化,但是读锁的版本号还没有更新。因此,在这种情况下,可能会出现一定的误判。

  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3197 引用 • 8215 回帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • WebComponents

    Web Components 是 W3C 定义的标准,它给了前端开发者扩展浏览器标签的能力,可以方便地定制可复用组件,更好的进行模块化开发,解放了前端开发者的生产力。

    1 引用 • 9 关注
  • BookxNote

    BookxNote 是一款全新的电子书学习工具,助力您的学习与思考,让您的大脑更高效的记忆。

    笔记整理交给我,一心只读圣贤书。

    1 引用 • 1 回帖 • 2 关注
  • 思源笔记

    思源笔记是一款隐私优先的个人知识管理系统,支持完全离线使用,同时也支持端到端加密同步。

    融合块、大纲和双向链接,重构你的思维。

    25203 引用 • 103945 回帖 • 1 关注
  • Chrome

    Chrome 又称 Google 浏览器,是一个由谷歌公司开发的网页浏览器。该浏览器是基于其他开源软件所编写,包括 WebKit,目标是提升稳定性、速度和安全性,并创造出简单且有效率的使用者界面。

    63 引用 • 289 回帖
  • CSS

    CSS(Cascading Style Sheet)“层叠样式表”是用于控制网页样式并允许将样式信息与网页内容分离的一种标记性语言。

    199 引用 • 542 回帖 • 1 关注
  • 爬虫

    网络爬虫(Spider、Crawler),是一种按照一定的规则,自动地抓取万维网信息的程序。

    106 引用 • 275 回帖 • 1 关注
  • BND

    BND(Baidu Netdisk Downloader)是一款图形界面的百度网盘不限速下载器,支持 Windows、Linux 和 Mac,详细介绍请看这里

    107 引用 • 1281 回帖 • 35 关注
  • Firefox

    Mozilla Firefox 中文俗称“火狐”(正式缩写为 Fx 或 fx,非正式缩写为 FF),是一个开源的网页浏览器,使用 Gecko 排版引擎,支持多种操作系统,如 Windows、OSX 及 Linux 等。

    7 引用 • 30 回帖 • 391 关注
  • Spark

    Spark 是 UC Berkeley AMP lab 所开源的类 Hadoop MapReduce 的通用并行框架。Spark 拥有 Hadoop MapReduce 所具有的优点;但不同于 MapReduce 的是 Job 中间输出结果可以保存在内存中,从而不再需要读写 HDFS,因此 Spark 能更好地适用于数据挖掘与机器学习等需要迭代的 MapReduce 的算法。

    74 引用 • 46 回帖 • 568 关注
  • golang

    Go 语言是 Google 推出的一种全新的编程语言,可以在不损失应用程序性能的情况下降低代码的复杂性。谷歌首席软件工程师罗布派克(Rob Pike)说:我们之所以开发 Go,是因为过去 10 多年间软件开发的难度令人沮丧。Go 是谷歌 2009 发布的第二款编程语言。

    498 引用 • 1395 回帖 • 249 关注
  • Kotlin

    Kotlin 是一种在 Java 虚拟机上运行的静态类型编程语言,由 JetBrains 设计开发并开源。Kotlin 可以编译成 Java 字节码,也可以编译成 JavaScript,方便在没有 JVM 的设备上运行。在 Google I/O 2017 中,Google 宣布 Kotlin 成为 Android 官方开发语言。

    19 引用 • 33 回帖 • 82 关注
  • Mobi.css

    Mobi.css is a lightweight, flexible CSS framework that focus on mobile.

    1 引用 • 6 回帖 • 757 关注
  • 外包

    有空闲时间是接外包好呢还是学习好呢?

    26 引用 • 233 回帖 • 1 关注
  • 酷鸟浏览器

    安全 · 稳定 · 快速
    为跨境从业人员提供专业的跨境浏览器

    3 引用 • 59 回帖 • 45 关注
  • FlowUs

    FlowUs.息流 个人及团队的新一代生产力工具。

    让复杂的信息管理更轻松、自由、充满创意。

    1 引用 • 1 关注
  • RYMCU

    RYMCU 致力于打造一个即严谨又活泼、专业又不失有趣,为数百万人服务的开源嵌入式知识学习交流平台。

    4 引用 • 6 回帖 • 55 关注
  • Sublime

    Sublime Text 是一款可以用来写代码、写文章的文本编辑器。支持代码高亮、自动完成,还支持通过插件进行扩展。

    10 引用 • 5 回帖 • 1 关注
  • Swift

    Swift 是苹果于 2014 年 WWDC(苹果开发者大会)发布的开发语言,可与 Objective-C 共同运行于 Mac OS 和 iOS 平台,用于搭建基于苹果平台的应用程序。

    36 引用 • 37 回帖 • 544 关注
  • Sillot

    Insights(注意当前设置 master 为默认分支)

    汐洛彖夲肜矩阵(Sillot T☳Converbenk Matrix),致力于服务智慧新彖乄,具有彖乄驱动、极致优雅、开发者友好的特点。其中汐洛绞架(Sillot-Gibbet)基于自思源笔记(siyuan-note),前身是思源笔记汐洛版(更早是思源笔记汐洛分支),是智慧新录乄终端(多端融合,移动端优先)。

    主仓库地址:Hi-Windom/Sillot

    文档地址:sillot.db.sc.cn

    注意事项:

    1. ⚠️ 汐洛仍在早期开发阶段,尚不稳定
    2. ⚠️ 汐洛并非面向普通用户设计,使用前请了解风险
    3. ⚠️ 汐洛绞架基于思源笔记,开发者尽最大努力与思源笔记保持兼容,但无法实现 100% 兼容
    29 引用 • 25 回帖 • 113 关注
  • 链书

    链书(Chainbook)是 B3log 开源社区提供的区块链纸质书交易平台,通过 B3T 实现共享激励与价值链。可将你的闲置书籍上架到链书,我们共同构建这个全新的交易平台,让闲置书籍继续发挥它的价值。

    链书社

    链书目前已经下线,也许以后还有计划重制上线。

    14 引用 • 257 回帖 • 1 关注
  • Sphinx

    Sphinx 是一个基于 SQL 的全文检索引擎,可以结合 MySQL、PostgreSQL 做全文搜索,它可以提供比数据库本身更专业的搜索功能,使得应用程序更容易实现专业化的全文检索。

    1 引用 • 222 关注
  • Flume

    Flume 是一套分布式的、可靠的,可用于有效地收集、聚合和搬运大量日志数据的服务架构。

    9 引用 • 6 回帖 • 652 关注
  • 设计模式

    设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软件开发人员经过相当长的一段时间的试验和错误总结出来的。

    200 引用 • 120 回帖 • 3 关注
  • InfluxDB

    InfluxDB 是一个开源的没有外部依赖的时间序列数据库。适用于记录度量,事件及实时分析。

    2 引用 • 87 关注
  • Notion

    Notion - The all-in-one workspace for your notes, tasks, wikis, and databases.

    10 引用 • 77 回帖
  • RESTful

    一种软件架构设计风格而不是标准,提供了一组设计原则和约束条件,主要用于客户端和服务器交互类的软件。基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存等机制。

    30 引用 • 114 回帖 • 5 关注
  • AWS
    11 引用 • 28 回帖 • 12 关注