高并发时为什么推荐ReentrantLock而不是synchronized
1、最初的 synchronized
它默认对临界资源添加重量级锁,即使可能并不存在竞争,只要走到临界区通通给你加锁。
现在来回答问题:
1) 如果是低于 JDK 1.5,抱歉你没得选,只能先将就着用 synchronized 重量级锁
2)如果你的 JDK 版本是1.5 那么推荐 ReentrantLock,这个时候的 synchronized 还是默认加重量级锁 。
接下来我们看看 JDK 1.6 干了啥
2、synchronized 的优化
在JDK 1.6 ,官方引入了如下的一系列优化:
- 锁消除
- 锁粗化
3)锁升级:无锁 -> 偏向锁 -> 轻量级锁 -> 重量级锁
重点介绍影响比较大的 "锁升级"
无锁: 不存在任何互斥资源的竞争,无需加任何锁。
偏向锁:当临界资源被某一个线程访问到了,那么在临界资源的对象头里标记偏向锁;在添加偏向锁的线程结束锁定前:1) 如果没有其它线程竞争该临界资源,效率等同无锁;2) 若此时另一个线程也希望访问该临界资源,那么偏向锁将升级为轻量级锁。(补充,新版本的JDK中,偏向锁已经被移除)
轻量级锁:何谓轻量级锁? 就是当被多个线程竞争的临界资源上,若存在轻量级锁时,等待获取锁的线程不会直接进入等待队列,而是先尝试指定次数的自旋,如果超过默认自旋次数仍未获取到锁,那么轻量级锁将会升级为重量级锁。(自旋次数JVM可配置,但是一般不建议动它)
重量级锁:如果临界资源上已经被某个线程,添加了重量级锁,那么当其它线程过来竞争锁的时候,就会直接进入等待队列,直至当前线程推出临界区,才会被唤醒、执行临界区。
然后我们重点说一下,轻量级锁的自旋,它好在哪里?了解JVM 并发编程模型的人都知道,在 JDK 21 LTS 推出虚拟线程之前,线程的调度是依托于底层的操作系统内核来完成的,相当于线程的:创建、等待、唤醒、销毁 ... 等等动作都直接依赖操作系统内核的API来完成,这是非常重型的操作, 所谓重量级锁,就是因为它直接阻塞别的线程,是一个非常耗费资源的操作。
而引入这个特性之后,其实 synchronized 的性能已经有了长足的进步,到了这里,其实跟 ReentrantLock 对比已经丝毫不怵了;甚至因为升级这个特性,有些时候 synchronized 会有更强悍的表现。
当 JDK1.6 <= version < JDK 21, 我们来回答问题:
高并发下 ReentrantLock 和 synchronized 哪个性能更好?
这个题可能没有标准答案,这里需要结合存在临界资源竞争的场景,来具体分析实际的临界操作行为,ReentrantLock 和 synchronized 谁更好。
- 当操作单一且耗时较短时,可以使用 synchronized 来做同步控制,因为只要不升级为重量级锁,性能损耗是极小的;且由于官方基于JVM做了针对性的优化, 那么它的适配性应当是更好的,如果没有复杂需求、临界操作并不复杂的时候,建议使用synchronized,因为它足够简单,性能也不在是鸿沟。
- 而 ReentrantLock它功能更加强大,它是在用户态下基于 AQS (Abstract Queued Synchornizer)实现,它不会阻塞内核线程。ReentrantLock 支持中断等待、支持公平锁、支持锁定多个对象,这些都是 synchronized 所不能具备的特性,在较为复杂的场景下,尤其建议使用 ReentrantLock 。
这里有一个值得思考的事情, AQS 实现的Lock 要考虑一个点,如果临界区很耗时,存在多个线程排队的时候,由于它没有进行上下文的切换,说明排队中的线程是没有被挂起的。在非虚拟线程场景下,jvm线程跟操作系统内核线程是一比一深度绑定的,那么是否应该指出这种情景下可能存在,cpu空转浪费资源。
不论是从ReentrantLock,还是synchronized的角度看,我们做程序设计时,应当保证临界区尽可能的小,尽可能的高效。
这里判断的临界值是:synchronized 阻塞再拉起线程的消耗,是否远大于临界区的执行时长。如果是,那么必须使用 Reentrantlock。否则,当阻塞再拉起线程的代价,远小于临界区执行的消耗时,应当慎重考虑使用何种锁。起码该JVM线程被阻塞后, 内核线程能释放出来干点别的事。
-- 如果有人,不看版本,不分析业务临界操作行为,就直接说谁好谁差,一律视为半桶水就行了。
3、但是,JAVA的最终答案 JDK 21 LTS 来了
JDK8 之后,最值得升级的 JDK 版本,虚拟线程,号称史诗级的加强,下一个主权就是它了!!!
自此,虚拟线程成为了JVM正式支持的功能.(jdk 19 的虚拟线程只是预览功能)
这里就不得不重点介绍一个特性了:JVM 并发编程模型,前文提到,在JDK 21 LTS 推出虚拟线程之前,jvm 的线程调度依赖操作系统内核。
在 JDK 21 的时代,JVM的线程被划分为了如下两种类型:
平台线程:所谓平台线程就是,跟操作系统内核深度绑定基于操作系统内核实现,其实它就是 JDK 21 LTS 版本之前的java 线程,"平台线程" 的调度依赖操作系统内核。
虚拟线程:JDK 21 LTS 新特性,"虚拟线程" 只在 JVM 内部调度,"虚拟线程"的调度,不再依赖操作系统内核的线程调度,它只发生在 JVM用户态,所以 "虚拟线程" 的调度行为是非常轻量的。"虚拟线程" 有等待队列,"虚拟线程"的执行需要靠"平台线程"来调度等待队列。可以理解为多个"虚拟线程",在竞争"平台线程" 的cpu时间分片.
在虚拟线程的场景下,我们开发者,调用的是虚拟线程,"平台线程"由 JVM自身进行控制。我们在 "虚拟线程" 内部不论执行了多少休眠、阻塞等操作,丝毫不会影响"平台线程"的调度。
- 当虚拟线程休眠、阻塞、等待时,那么它将会从 "平台线程" 上 unmount 并进入等待队列
- 当 "平台线程" 上unmount 了某个虚拟线程,或者某个虚拟线程执行结束了,那么 "平台线程" 会在依据一定的规则,从虚拟线程等待队列里唤起并mount某一个虚拟线程,执行时间分片。
- 虚拟线程所需的空间极小,在 JDK 21 环境下,可以清理拉起数万、甚至数十万的虚拟线程,同等硬件资源下,相较于之前的并发编程模型,可以更多拉起的线程数,已经到指数级了。这是几近可以跟 golang 在高并发领域,掰头一下的水平了,这够高并发了吧?
由此来看 "虚拟线程" 这种用户态实现的并发编程模型,在高并发场景下,要远胜于之前的 "操作系统内核态" 并发编程模型。
我用了这么多文字介绍JVM并发编程模型,那么它跟 ReentrantLock 和 synchronized 有什么关系呢?
这是因为,synchronized 会将虚拟线程固定在平台线程上,在 synchronized 临界区执行结束之前,无法被 unmount, 这样是会影响高并发场景下虚拟线程的调度效率的。
而ReentrantLock 就不会有这样的问题了,碰到阻塞操作时,"虚拟线程"会很丝滑的从"平台线程"上unmount,让出"平台线程的"时间分片。
现在我们基于 JDK 21 LTS 版本来回答问题:高并发下 ReentrantLock 和 synchronized 哪个性能更好?
1)首先为了有更好的高并发体验(IO密集型),应当使用 "虚拟线程"进行开发
2) 由于 synchronized 会阻止虚拟线程遇到阻塞后从 "平台线程"上 卸载,所以不推荐在高并发场景使用它,它会极大的影响虚拟线程在平台线程上的调度。在虚拟线程的版本, "ReentrantLock" 才是最终答案。
当然你要是说我不用 "虚拟线程" ,那答案同第二节所述。
高并发时为什么推荐ReentrantLock而不是synchronized的更多相关文章
- 关于sphinx+PHP在高并发时响应性能低下的解决办法
经过多次压力测试,发现sphinx在高并发时出现负载突然提升,并且响应速度明显下降.经过多方面的排查,发现是由于PHP与sphinx自带的 searchd进行socket的连接之后,系统内存有大量的T ...
- [转]你如何面对—LNMP高并发时502
From : http://www.topthink.com/topic/5683.html 之前php-fpm配置: 单个php-fpm实例,使用socket方式,内存8G 静态方式,启动php-f ...
- j2ee高并发时使用全局变量需要注意的问题
原文:https://blog.csdn.net/jston_learn/article/details/21617311 开发中,全局变量的使用很频繁,但对于多线程的访问,使用全局变量需要注意的地方 ...
- php session在高并发时可能存在的问题。
如果同一个客户端并发发送多个请求,而每个请求都使用了Session,那么PHP Session锁的存在会导致服务器串行响应这些请求,而不是并行.这是因为在默认情况下,PHP使用文件存储Session数 ...
- 【多线程与高并发原理篇:4_深入理解synchronized】
1. 前言 越是简单的东西,在深入了解后发现越复杂.想起了曾在初中阶段,语文老师给我们解说<论语>的道理,顺便给我们提了一句,说老子的无为思想比较消极,学生时代不要太关注.现在有了一定的生 ...
- 并发与高并发(八)-线程安全性-原子性-synchronized
前言 闲暇时刻,谈一下曾经在多线程教程中接触的同步锁synchronized,相当于复习一遍吧. 主要介绍 synchronized:依赖JVM Lock:依赖特殊的CPU指令,代码实现,Reetra ...
- 《JAVA高并发编程详解》-volatile和synchronized
- java高并发_博客-网址-资料 推荐
大概说一下自己作为入门学习java高并发的博客地址,很不错在自己的博客里记录一下:如果能有刷到我的博客的骚年,又刚好想了解java高并发,强烈推荐看看 地址:http://www.itsoku.com ...
- Redis 高并发带来的一些问题
前言 本文讲述Redis在遇到高并发时的一些问题.即遇到大量请求时需要思考的点,如缓存穿透 缓存击穿 缓存雪崩 热key处理.一般中小型传统软件企业,很难碰到这个问题.如果有大并发的项目,流量有几百万 ...
- Redis高并发分布式锁详解
为什么需要分布式锁 1.为了解决Java共享内存模型带来的线程安全问题,我们可以通过加锁来保证资源访问的单一,如JVM内置锁synchronized,类级别的锁ReentrantLock. 2.但是随 ...
随机推荐
- IO调度算法的简单学习与整理
IO调度算法的简单学习与整理 前言 前几天整理了 /sys/block/sda/queue/nr_requests 以及 /sys/block/sda/device/queue_depth 的两个参数 ...
- [转帖]15 个必须知道的 chrome 开发工具技巧
在Web开发者中,Google Chrome是使用最广泛的浏览器.六周一次的发布周期和一套强大的不断扩大开发功能,使其成为了web开发者必备的工具.你可能已经熟悉了它的部分功能,如使用console和 ...
- Systemd设置ulimit的方式与方法
Systemd设置ulimit的方式与方法 摘要 Linux安装完成之后前面几件事情一般是处理selinux 以及处理ulimit 其实处理文件打开数有多种方法,之前也总结过, 但是最近因为syste ...
- It is currently in use by another Gradle instance
FAILURE: Build failed with an exception. * What went wrong: Could not create service of type TaskHis ...
- 【小测试】玩一玩 VictoriaMetrics 的 force merge
作者:张富春(ahfuzhang),转载时请注明作者和引用链接,谢谢! cnblogs博客 zhihu Github 公众号:一本正经的瞎扯 我是期望通过备份来建立 VictoriaMetrics 的 ...
- 【JS 逆向百例】房天下登录接口参数逆向
声明 本文章中所有内容仅供学习交流,严禁用于商业用途和非法用途,否则由此产生的一切后果均与作者无关,若有侵权,请联系我立即删除! 逆向目标 目标:房天下账号密码登录 主页:https://passpo ...
- 6.1 C/C++ 封装字符串操作
C/C++语言是一种通用的编程语言,具有高效.灵活和可移植等特点.C语言主要用于系统编程,如操作系统.编译器.数据库等:C语言是C语言的扩展,增加了面向对象编程的特性,适用于大型软件系统.图形用户界面 ...
- Python 实现 WebSocket 通信
WebSocket 协议主要用于解决Web前端与后台数据交互问题,在WebSocket技术没有被定义之前,前台与后端通信需要使用轮询的方式实现,WebSocket则是通过握手机制让客户端与服务端建立全 ...
- Eslint 的rules一些配置 (.eslintrc.js文件中的rules选项)
rules: { // off=0, warn=1, error=2, 如果是数组, 第二项表示参数option // indent: [2, 2], // 控制缩进为2 eqeqeq: 1, // ...
- Exception message: CreateSymbolicLink error (1314): ???????????
window下运行任务报错:Exception message: CreateSymbolicLink error (1314): ??????????? 报错信息如下: Diagnostics: E ...