Java 8 中 ReentrantLock 与 Synchronized 的区别

🚀 作者主页: 有来技术
🔥 开源项目: youlai-mall 🍃 vue3-element-admin 🍃 youlai-boot
🌺 仓库主页: Gitee 💫 Github 💫 GitCode
💖 欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请纠正!

目录

  • 引言
  • 同步机制简介
    • Synchronized
    • ReentrantLock
  • 区别分析
    • 可中断性
    • 公平性
    • 条件变量
    • 性能比较
  • Java 8 的新特性
  • 面试题解析
    • 什么时候使用 ReentrantLock 而不是 synchronized?
    • ReentrantLock 的可重入性是什么意思?
    • 什么是乐观读锁?StampedLock 有什么优势?
  • 总结

引言

在Java多线程编程中,确保线程安全是至关重要的。为了实现线程的同步和协作,Java提供了多种同步机制,其中最常见的是使用 synchronized 关键字和 ReentrantLock 类。这两者在实现线程安全的同时,也存在着一些区别。本文将深入讨论这两种同步机制的差异,并结合 Java 8 的特性,解析它们在常见面试题中的应用。

同步机制简介

Synchronized

在这里插入图片描述

synchronized 是Java中最早引入的同步机制,通过对方法或代码块加锁来确保多线程环境下的数据一致性。它使用起来简单,不需要手动释放锁,JVM会自动管理。

public synchronized void synchronizedMethod() {// 同步代码块// ...
}

ReentrantLock

ReentrantLock 是Java 5 中引入的同步机制,相较于 synchronized,它提供了更灵活的锁定方式。通过显式地获取锁和释放锁,程序员可以更精细地控制同步操作。此外,ReentrantLock 还支持可重入性,即一个线程可以多次获取同一个锁。

import java.util.concurrent.locks.ReentrantLock;public class MyLock {private final ReentrantLock lock = new ReentrantLock();public void lockedMethod() {lock.lock();try {// 同步代码块// ...} finally {lock.unlock();}}
}

区别分析

可中断性

一个显著的区别是 ReentrantLock 支持中断等待锁的线程,而 synchronized 不支持。在 ReentrantLock 中,线程可以通过 lockInterruptibly 方法等待锁,如果其他线程中断了当前线程,它可以响应中断而不是一直等待。

ReentrantLock lock = new ReentrantLock();public void interruptibleMethod() throws InterruptedException {lock.lockInterruptibly();try {// 同步代码块// ...} finally {lock.unlock();}
}

公平性

ReentrantLock 可以选择是否公平地获取锁,而 synchronized 是非公平的。在公平模式下,等待时间最长的线程会优先获得锁。在非公平模式下,线程有一定几率在等待队列为空时插队成功。

ReentrantLock fairLock = new ReentrantLock(true); // 公平锁
ReentrantLock unfairLock = new ReentrantLock();     // 非公平锁

条件变量

ReentrantLock 提供了 Condition 接口,可以通过 newCondition 方法创建多个条件变量,用于在不同的情况下等待或唤醒线程。而 synchronized 只能通过 ObjectwaitnotifynotifyAll 方法来实现简单的线程协作,缺乏 ReentrantLock 中的灵活性。

ReentrantLock lock = new ReentrantLock();
Condition condition = lock.newCondition();public void awaitSignal() throws InterruptedException {lock.lock();try {condition.await();} finally {lock.unlock();}
}public void sendSignal() {lock.lock();try {condition.signal();} finally {lock.unlock();}
}

性能比较

在 Java 5 之前,synchronized 的性能较差,但在 Java 6 及以后的版本中,对 synchronized 进行了优化,性能已经有了很大的提升。一般来说,在性能上,synchronizedReentrantLock 的差距并不明显。在选择使用时,更应该考虑到代码的可读性和维护性。

Java 8 的新特性

Java 8 引入了函数式编程的特性,同时也对并发编程做出了一些改进。其中一个显著的改变是引入了新的 StampedLock 类,它是 ReentrantLock 的进一步扩展,提供了乐观锁机制,使得在读多写少的场景下性能更优。

import java.util.concurrent.locks.StampedLock;public class MyStampedLock {private final StampedLock lock = new StampedLock();public void read() {long stamp = lock.tryOptimisticRead();// 乐观读操作if (!lock.validate(stamp)) {// 如果发生了写操作,则使用悲观读锁stamp = lock.readLock();try {// 悲观读操作} finally {lock.unlockRead(stamp);}}}public void write() {long stamp = lock.writeLock();try {// 写操作} finally {lock.unlockWrite(stamp);}}
}

面试题解析

什么时候使用 ReentrantLock 而不是 synchronized?

  • 需要支持可中断等待: 如果你的线程需要响应中断而不是一直等待锁,那么 ReentrantLock 是更好的选择。
  • 需要尝试获取锁: ReentrantLock 提供了 tryLock 方法,可以尝试获取锁而不一直等待,可以用于避免死锁。
  • 需要选择公平性: 如果你需要控制线程获取锁的顺序,可以选择 ReentrantLock 的公平锁或非公平锁。

ReentrantLock 的可重入性是什么意思?

可重入性是指同一个线程在持有锁的情况下,能够再次获取这个锁而不发生死锁。ReentrantLocksynchronized 都是可重入的,这使得线程可以递归地调用同步方法或同步代码块。

什么是乐观读锁?StampedLock 有什么优势?

乐观读锁是 StampedLock 引入的特性,它允许多个线程同时读取共享资源,而不会阻塞写锁。如果在乐观读锁期间没有写锁被获取,读操作可以立即进行。当然,如果写锁被获取,就需要转为悲观读锁。

StampedLock 的优势在于在读多写少的场景下性能更优,因为它允许多个线程同时读取,而不阻塞彼此。

总结

在Java 8中,ReentrantLocksynchronized 依然是同步机制中的主流选择。选择使用哪一种取决于具体的需求和场景。总体而言,synchronized 在简单场景下易用性更高,而 ReentrantLock 则在一些复杂场景下提供了更多的灵活性和控制权。

对于面试而言,了解这两者的区别、使用场景以及新特性,可以展现出对多线程编程深入理解的能力,为应对面试中的技术问题提供有力支持。在实际应用中,结合业务需求和性能要求,选择适当的同步机制是至关重要的。

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

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

相关文章

什么是轻量应用服务器?可以从亚马逊云科技的优势入手了解

什么是轻量应用服务器? 随着如今各行各业对云计算的需求越来越多,云服务器也被越来越多的企业所广泛采用。其中,轻量应用服务器是一种简单、高效、可靠的云计算服务,能够为开发人员、企业和个人提供轻量级的虚拟专用服务器&#x…

Python语言学习笔记之二(基础语法)

本课程对于有其它语言基础的开发人员可以参考和学习,同时也是记录下来,为个人学习使用,文档中有此不当之处,请谅解。 Python几种字符串的表示: 在Python中,字符串是一种基本的数据类型,可以使…

【shell】多行重定向与免交互expect与ssh、scp的结合使用

目录 一、多行重定向 举例1:使用read命令接收用户的输入值会有交互过程 举例2:设置变量的值 举例3:创建用户密码 举例4:使用多行重定向写入文件中(以repo文件举例) 举例5:变量设定 二、免…

解析d3dcompiler_47.dll缺失怎么修复,4种方法修复d3dcompiler_47.dll文件

d3dcompiler_47.dll缺失怎么修复?其实在我们使用计算机操作的过程中,有时会遇到一些由dll文件错误导致的问题,其中d3dcompiler_47.dll丢失就是这样一种。那么究竟d3dcompiler_47.dll缺失是什么意思,为何它会发生丢失,以…

WebGL笔记:图形旋转的原理和实现

旋转 1 )旋转的概念 三维物体的旋转要比位移复杂一点,三维物体的旋转需要满足以下条件: 旋转轴旋转方向旋转角度 场景举例 模型站在旋转轴的起点进行旋转模型要往左转还是往右转,就是旋转的方向模型旋转的大小就是旋转角度 2 &…

HarmonyOS应用开发实战—登录页面【ArkTS】

文章目录 本页面实战效果预览图一.HarmonyOS应用开发1.1HarmonyOS 详解1.2 ArkTS详解二.HarmonyOS应用开发实战—登录页面【ArkTS】2.1 ArkTS页面源码2.2 代码解析2.3 心得本页面实战效果预览图 一.HarmonyOS应用开发 1.1HarmonyOS 详解 HarmonyOS(鸿蒙操作系统)是华为公司…

【计算机网络笔记】以太网

系列文章目录 什么是计算机网络? 什么是网络协议? 计算机网络的结构 数据交换之电路交换 数据交换之报文交换和分组交换 分组交换 vs 电路交换 计算机网络性能(1)——速率、带宽、延迟 计算机网络性能(2)…

Kafka事务机制:原理和实践

Kafka事务机制:原理和实践 Apache Kafka 是一个分布式流处理平台,广泛用于构建实时数据管道和流应用程序。它不仅以高吞吐量、可扩展性和容错能力著称,还提供了事务支持,以确保数据的完整性和一致性。在这篇博客中,我…

现代C++之万能引用、完美转发、引用折叠

现代C之万能引用、完美转发、引用折叠 0.导语1.问题引入2.引入万能引用3.万能引用出现场合4.理解左值与右值4.1 精简版4.2 完整版4.3 生命周期延长4.4 生命周期延长应用5.区分万能引用6.表达式的左右值性与类型无关7.引用折叠和完美转发7.1 引用折叠之本质细节7.2 示例与使用7.…

【libGDX】Mesh纹理贴图

1 前言 纹理贴图的本质是将图片的纹理坐标与模型的顶点坐标建立一一映射关系。纹理坐标的 x、y 轴正方向分别朝右和朝下,如下。 2 纹理贴图 本节将使用 Mesh、ShaderProgram、Shader 实现纹理贴图,OpenGL ES 的实现见博客 → 纹理贴图,本节完…

excel表格在线编辑(开源版)

文章目录 前言一、Luckysheetvue3vite 例子如有启发,可点赞收藏哟~ 前言 本文记录好用的开源在线表格 具体如图显示 另外记录下更名后的univer~,如下图(有兴趣可自行详细了解) univer 在线思维导图 一、Luckysheet 参考git…

微服务--05--配置管理

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 配置管理这些问题都可以通过统一的配置管理器服务解决。而Nacos不仅仅具备注册中心功能,也具备配置管理的功能: 1.配置共享1.1.添加共享配置…

C#,数值计算——插值和外推,径向基函数插值(RBF_inversemultiquadric)的计算方法与源程序

1 文本格式 using System; namespace Legalsoft.Truffer { public class RBF_inversemultiquadric : RBF_fn { private double r02 { get; set; } public RBF_inversemultiquadric(double scale 1.0) { this.r02 Globals.SQR(scale); …

鸿蒙4.0开发笔记之ArkTS语法的基础数据类型[DevEco Studio开发](七)

文章目录 一、基本数据类型的定义1、变量声明2、数字类型3、字符串类型4、布尔类型5、数组类型6、元组类型7、枚举类型8、联合类型(少用)9、未知Unkown类型10、未定义和空值类型 二、数据类型的使用1、组件内部声明变量不需要使用let关键字2、使用Divide…

【数据结构】树与二叉树(廿四):树搜索指定数据域的结点(算法FindTarget)

文章目录 5.3.1 树的存储结构5. 左儿子右兄弟链接结构 5.3.2 获取结点的算法1. 获取大儿子、大兄弟结点2. 搜索给定结点的父亲3. 搜索指定数据域的结点a. 算法FindTargetb. 算法解析c. 代码实现a. 使用指向指针的指针b. 直接返回找到的节点 4. 代码整合 5.3.1 树的存储结构 5.…

论文阅读 Forecasting at Scale (二)

最近在看时间序列的文章,回顾下经典 论文地址 项目地址 Forecasting at Scale 3.2、季节性 3.3、假日和活动事件3.4、模型拟合3.5、分析师参与的循环建模4、自动化预测评估4.1、使用基线预测4.2、建模预测准确性4.3、模拟历史预测4.4、识别大的预测误差 5、结论6、致…

前后端分离开发出现的跨域问题

先说说什么是跨域。 请求的URL地址中的协议、域名、端口号中的任意一个与当前URL不同就是跨域。 比如: 当前页面的URL请求的URL是否跨域原因htttp://localhost:8080htttps://localhost:8080是协议不同htttp://localhostll:8080htttp://localhost:8080是域名不同htt…

计算机网络高频面试八股文

目录: 网络分层结构三次握手两次握手可以吗?四次挥手第四次挥手为什么要等待2MSL?为什么是四次挥手?TCP有哪些特点?说说TCP报文首部有哪些字段,其作用又分别是什么?TCP和UDP的区别?…

IWDG和WWDG HAL库+cubeMX

一.IWDG 1.原理 启用IWDG后,LSI时钟会自动开启 2.IWDG溢出时间计算 3.IWDG配置步骤 4.HAL库相关函数介绍 HAL_IWDG_Init //使能IWDG,设置预分频系数和重装载值等 HAL_IWDG_Refresh //把重装载寄存器的值重载到计数器中,喂狗typedef str…

【C++ 程序设计入门基础】- 第3节-循环结构01

目录 循环结构 一、for 语句 for 循环案例 输入一个整数n,输出1~n的所有整数。 编译运行,查看输出结果 编译调试 for 循环结构语义分析 二、beak 语句 三、continue 语句 案例1: 案例2: 案例3: 循环…