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

Java锁如何优化

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

北京

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

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

看不清楚,换张图片

免费获取短信验证码

Java锁如何优化

这篇文章主要介绍“Java锁如何优化”,在日常操作中,相信很多人在Java锁如何优化问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java锁如何优化”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

  锁优化

  这里的锁优化主要是指 JVM 对 synchronized 的优化。

  自旋锁

  互斥同步进入阻塞状态的开销都很大,应该尽量避免。在许多应用中,共享数据的锁定状态只会持续很短的一段时间。自旋锁的思想是让一个线程在请求一个共享数据的锁时执行忙循环(自旋)一段时间,如果在这段时间内能获得锁,就可以避免进入阻塞状态。

  自旋锁虽然能避免进入阻塞状态从而减少开销,但是它需要进行忙循环操作占用 CPU 时间,它只适用于共享数据的锁定状态很短的场景。

  在 JDK 1.6 中引入了自适应的自旋锁。自适应意味着自旋的次数不再固定了,而是由前一次在同一个锁上的自旋次数及锁的拥有者的状态来决定。

  锁消除

  锁消除是指对于被检测出不可能存在竞争的共享数据的锁进行消除。

  锁消除主要是通过逃逸分析来支持,如果堆上的共享数据不可能逃逸出去被其它线程访问到,那么就可以把它们当成私有数据对待,也就可以将它们的锁进行消除。

  对于一些看起来没有加锁的代码,其实隐式的加了很多锁。例如下面的字符串拼接代码就隐式加了锁:

  public static String concatString(String s1, String s2, String s3) { return s1 + s2 + s3; }

  String 是一个不可变的类,编译器会对 String 的拼接自动优化。在 JDK 1.5 之前,会转化为 StringBuffer 对象的连续 append() 操作:

  public static String concatString(String s1, String s2, String s3) { StringBuffer sb = new StringBuffer(); sb.append(s1); sb.append(s2); sb.append(s3); return sb.toString(); }

  每个 append() 方法中都有一个同步块。虚拟机观察变量 sb,很快就会发现它的动态作用域被限制在 concatString() 方法内部。也就是说,sb 的所有引用永远不会逃逸到 concatString() 方法之外,其他线程无法访问到它,因此可以进行消除。

  锁粗化

  如果一系列的连续操作都对同一个对象反复加锁和解锁,频繁的加锁操作就会导致性能损耗。

  上一节的示例代码中连续的 append() 方法就属于这类情况。如果虚拟机探测到由这样的一串零碎的操作都对同一个对象加锁,将会把加锁的范围扩展(粗化)到整个操作序列的外部。对于上一节的示例代码就是扩展到第一个 append() 操作之前直至最后一个 append() 操作之后,这样只需要加锁一次就可以了。

  轻量级锁

  JDK 1.6 引入了偏向锁和轻量级锁,从而让锁拥有了四个状态:无锁状态(unlocked)、偏向锁状态(biasble)、轻量级锁状态(lightweight locked)和重量级锁状态(inflated)。

  重量级锁也就是通常说synchronized的对象锁。

  以下是 HotSpot 虚拟机对象头的内存布局,这些数据被称为 Mark Word。其中 tag bits 对应了五个状态,这些状态在右侧的 state 表格中给出。除了 marked for gc 状态,其它四个状态已经在前面介绍过了。

  下图左侧是一个线程的虚拟机栈,其中有一部分称为 Lock Record 的区域,这是在轻量级锁运行过程创建的,用于存放锁对象的 Mark Word。而右侧就是一个锁对象,包含了 Mark Word 和其它信息。

  轻量级锁是相对于传统的重量级锁而言,它使用 CAS 操作来避免重量级锁使用互斥量的开销。对于绝大部分的锁,在整个同步周期内都是不存在竞争的,因此也就不需要都使用互斥量进行同步,可以先采用 CAS 操作进行同步,如果 CAS 失败了再改用互斥量进行同步。

  当尝试获取一个锁对象时,如果锁对象标记为 0 01,说明锁对象的锁未锁定(unlocked)状态。此时虚拟机在当前线程的虚拟机栈中创建 Lock Record,然后使用 CAS 操作将对象的 Mark Word 更新为 Lock Record 指针。如果 CAS 操作成功了,那么线程就获取了该对象上的锁,并且对象的 Mark Word 的锁标记变为 00,表示该对象处于轻量级锁状态。

  如果 CAS 操作失败了,虚拟机首先会检查对象的 Mark Word 是否指向当前线程的虚拟机栈,如果是的话说明当前线程已经拥有了这个锁对象,那就可以直接进入同步块继续执行,否则说明这个锁对象已经被其他线程线程抢占了。如果有两条以上的线程争用同一个锁,那轻量级锁就不再有效,要膨胀为重量级锁。

  偏向锁

  偏向锁的思想是偏向于第一个获取锁对象的线程,这个线程在之后获取该锁就不再需要进行同步操作,甚至连 CAS 操作也不再需要。

  当锁对象第一次被线程获得的时候,进入偏向状态,标记为 1 01。同时使用 CAS 操作将线程 ID 记录到 Mark Word 中,如果 CAS 操作成功,这个线程以后每次进入这个锁相关的同步块就不需要再进行任何同步操作。

  当有另外一个线程去尝试获取这个锁对象时,偏向状态就宣告结束,此时撤销偏向(Revoke Bias)后恢复到未锁定状态或者轻量级锁状态。

到此,关于“Java锁如何优化”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

免责声明:

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

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

Java锁如何优化

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

下载Word文档

猜你喜欢

如何优化mysql行锁

如何优化mysql行锁?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。1、优化方法尽可能让所有数据检索都通过索引来完成,避免无索引行或索引失效导致行锁升级为表锁。尽可能避免间
2023-06-15

如何优化MySQL的INSERT锁

要优化MySQL的INSERT锁,可以考虑以下几个方面:使用合适的存储引擎:不同的存储引擎对INSERT操作的锁处理方式不同。InnoDB存储引擎支持行级锁,可以减少锁冲突的概率,提高并发性能。如果表的读操作较多,可以考虑使用MyISAM存
如何优化MySQL的INSERT锁
2024-08-14

浅谈Java的Synchronized锁原理和优化

这篇文章主要介绍了Java的Synchronized锁原理和优化,synchronized的作用是保证在同一时刻,被修饰的代码块或方法只会有一个线程执行,以达到保证并发安全的效果,需要的朋友可以参考下
2023-05-20

如何优化Java虚拟机

这篇文章将为大家详细讲解有关如何优化Java虚拟机,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。堆设置-Xmx3550m:设置JVM最大堆内存为3550M。-Xms3550m:设置JVM初始堆内存为355
2023-05-30

如何实现MySQL底层优化:事务锁的性能优化和避免死锁的方法

如何通过事务锁的性能优化和避免死锁来实现MySQL底层优化导言:在MySQL数据库中,事务锁起着至关重要的作用。如果事务锁的性能不好或者存在死锁,将严重影响数据库的性能和稳定性。因此,本文将重点介绍如何通过优化事务锁的性能和避免死锁来实现M
如何实现MySQL底层优化:事务锁的性能优化和避免死锁的方法
2023-11-08

MySQL INSERT锁与锁升级优化

MySQL中的INSERT锁是一种特殊的行级锁,用于在插入新记录时保护数据表的一致性。INSERT锁会在插入新记录时自动加上,并在事务提交或回滚时释放。同时,MySQL还会根据需要将INSERT锁升级为更高级别的锁,如表级锁或页级锁,以提高
MySQL INSERT锁与锁升级优化
2024-08-19

java接口性能如何优化

优化Java接口的性能可以从以下几个方面入手:1. 减少接口方法数量:接口方法越多,调用时的开销就越大。因此,可以考虑将一些耗时较长的方法进行合并或者拆分为多个小的方法。2. 接口设计合理化:合理的接口设计可以减少不必要的方法调用,从而提高
2023-08-20

如何实现MySQL底层优化:事务锁的高级性能优化和避免死锁的方法

如何实现MySQL底层优化:事务锁的高级性能优化和避免死锁的方法引言:在数据库系统中,事务锁是保证数据一致性和并发访问的重要机制之一。然而,在高并发场景下,事务锁可能会导致性能问题和死锁。为了提高MySQL性能,我们需要对事务锁进行高级性能
如何实现MySQL底层优化:事务锁的高级性能优化和避免死锁的方法
2023-11-08

处理和优化 MySQL 死锁锁定

很抱歉,由于技术限制,我无法提供具体代码示例。但是我可以帮你提供一个讲解MySQL锁、死锁处理与优化的大纲,供你参考。MySQL 锁的死锁处理与优化一、MySQL锁的分类读锁(共享锁)写锁(排它锁)二、死锁概念什么是死锁死锁产生的条件如何避
处理和优化 MySQL 死锁锁定
2023-12-21

java多层if嵌套如何优化

在Java中,多层的if嵌套可以通过以下方式进行优化:1. 使用逻辑运算符来简化条件判断:可以使用逻辑与(&&)和逻辑或(||)运算符来组合条件判断,从而减少嵌套的层数。2. 使用switch语句代替多层if嵌套:如果条件判断是基于一个变量
2023-09-13

java中如何优化大量if...else...

本文小编为大家详细介绍“java中如何优化大量if...else...”,内容详细,步骤清晰,细节处理妥当,希望这篇“java中如何优化大量if...else...”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。策
2023-07-05

log4j2的高并发死锁问题配置如何优化

这篇文章给大家分享的是有关log4j2的高并发死锁问题配置如何优化的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。log4j2高并发死锁问题配置优化Maven中pom.xml引用2.
2023-06-22

优化与调优MySQL的锁机制

MySQL 锁的优化与调优在高并发的数据库操作中,锁是非常重要的机制之一。MySQL 提供了各种类型的锁,如共享锁、排他锁、表锁、行锁等,来保证数据的一致性和并发控制。然而,在大规模的数据库应用中,锁也可能成为性能瓶颈,影响系统的吞吐能力。
优化与调优MySQL的锁机制
2023-12-21

编程热搜

目录