返回顶部
首页 > 资讯 > 前端开发 > JavaScript >Java锁如何优化
  • 525
分享到

Java锁如何优化

2024-04-02 19:04:59 525人浏览 八月长安
摘要

这篇文章主要介绍“Java锁如何优化”,在日常操作中,相信很多人在Java锁如何优化问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”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锁如何优化”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

--结束END--

本文标题: Java锁如何优化

本文链接: https://lsjlt.com/news/93996.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Java锁如何优化
    这篇文章主要介绍“Java锁如何优化”,在日常操作中,相信很多人在Java锁如何优化问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java锁如何优化”的疑惑有所帮助!接下来,...
    99+
    2024-04-02
  • 如何优化mysql行锁
    如何优化mysql行锁?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。1、优化方法尽可能让所有数据检索都通过索引来完成,避免无索引行或索引失效导致行锁升级为表锁。尽可能避免间...
    99+
    2023-06-15
  • DB2性能优化 – 如何通过调整锁参数优化锁升级
    1、概念描述所谓的锁升级(lock escalation),是数据库的一种作用机制,为了节约内存的开销, 其会将为数众多并占用大量资源的细粒度的锁转化为数量较少的且占用相对较少资源的粗粒度的锁,多数情况下主...
    99+
    2024-04-02
  • java自旋锁和JVM对锁的优化详解
    目录背景好处AtomicLong的实现getAndIncrement方法实验缺点适用场景JVM对锁做了哪些优化?自适应的自旋锁锁消除锁粗化偏向锁/ 轻量级锁/ 重量级锁锁升级背景 先...
    99+
    2024-04-02
  • 浅谈Java的Synchronized锁原理和优化
    目录一、synchronized介绍二、synchronized的使用1.修饰方法三、synchronized的底层实现对象头监视器(Monitor)四、synchronized 锁...
    99+
    2023-05-20
    Java Synchronized锁 Synchronized锁原理 Synchronized锁优化
  • 如何优化Java虚拟机
    这篇文章将为大家详细讲解有关如何优化Java虚拟机,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。堆设置-Xmx3550m:设置JVM最大堆内存为3550M。-Xms3550m:设置JVM初始堆内存为355...
    99+
    2023-05-30
  • 如何实现MySQL底层优化:事务锁的性能优化和避免死锁的方法
    如何通过事务锁的性能优化和避免死锁来实现MySQL底层优化导言:在MySQL数据库中,事务锁起着至关重要的作用。如果事务锁的性能不好或者存在死锁,将严重影响数据库的性能和稳定性。因此,本文将重点介绍如何通过优化事务锁的性能和避免死锁来实现M...
    99+
    2023-11-08
    MySQL 优化 死锁 事务锁
  • Java 中如何优化 load 性能?
    在 Java 程序中,我们经常需要从磁盘或网络中读取数据。在这种情况下,load 操作的性能是非常关键的,因为它直接影响了整个程序的运行效率。本文将介绍一些优化 load 性能的技巧,帮助你提高程序的性能表现。 使用缓存 缓存是提高 ...
    99+
    2023-10-15
    load spring 编程算法
  • java接口性能如何优化
    优化Java接口的性能可以从以下几个方面入手:1. 减少接口方法数量:接口方法越多,调用时的开销就越大。因此,可以考虑将一些耗时较长...
    99+
    2023-08-20
    java
  • 处理和优化 MySQL 死锁锁定
    很抱歉,由于技术限制,我无法提供具体代码示例。但是我可以帮你提供一个讲解MySQL锁、死锁处理与优化的大纲,供你参考。MySQL 锁的死锁处理与优化一、MySQL锁的分类读锁(共享锁)写锁(排它锁)二、死锁概念什么是死锁死锁产生的条件如何避...
    99+
    2023-12-21
    MySql锁 死锁处理 锁优化
  • 如何实现MySQL底层优化:事务锁的高级性能优化和避免死锁的方法
    如何实现MySQL底层优化:事务锁的高级性能优化和避免死锁的方法引言:在数据库系统中,事务锁是保证数据一致性和并发访问的重要机制之一。然而,在高并发场景下,事务锁可能会导致性能问题和死锁。为了提高MySQL性能,我们需要对事务锁进行高级性能...
    99+
    2023-11-08
    死锁 MySQL底层优化关键词:事务 锁优化
  • 详解Java编译优化之循环展开和粗化锁
    目录循环展开和粗化锁分析Assembly日志禁止Loop unrolling循环展开和粗化锁 我们先来回顾一下什么是循环展开。 循环展开就是说,像下面的循环遍历的例子: for ...
    99+
    2024-04-02
  • Java synchronized底层实现原理以及锁优化
    目录一、概述synchronized简介synchronized作用synchronized的使用二、实现原理三、理解Java对象头四、JVM对synchronized的锁优化1、偏...
    99+
    2024-04-02
  • java多层if嵌套如何优化
    在Java中,多层的if嵌套可以通过以下方式进行优化:1. 使用逻辑运算符来简化条件判断:可以使用逻辑与(&&)和逻辑或(||)运算...
    99+
    2023-09-13
    java
  • java中如何优化大量if...else...
    本文小编为大家详细介绍“java中如何优化大量if...else...”,内容详细,步骤清晰,细节处理妥当,希望这篇“java中如何优化大量if...else...”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。策...
    99+
    2023-07-05
  • 优化与调优MySQL的锁机制
    MySQL 锁的优化与调优在高并发的数据库操作中,锁是非常重要的机制之一。MySQL 提供了各种类型的锁,如共享锁、排他锁、表锁、行锁等,来保证数据的一致性和并发控制。然而,在大规模的数据库应用中,锁也可能成为性能瓶颈,影响系统的吞吐能力。...
    99+
    2023-12-21
    MySQL优化 锁优化 锁调优
  • log4j2的高并发死锁问题配置如何优化
    这篇文章给大家分享的是有关log4j2的高并发死锁问题配置如何优化的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。log4j2高并发死锁问题配置优化Maven中pom.xml引用<log4j2.version...
    99+
    2023-06-22
  • Java中的synchronized 优化方法之锁膨胀机制
    目录synchronized什么是用户态和内核态?为什么分内核态和用户态?锁膨胀偏向锁偏向锁执行流程偏向锁的优点Mark Word 扩展知识:内存布局轻量级锁注意事项重量级锁总结前言...
    99+
    2024-04-02
  • MySQL InnoDB行锁优化建议
    InnoDB存储引擎由于实现了行级锁定,虽然在锁定机制的实现方面所带来的性能损耗可能比表级锁定会更高一些,但是在整体并发处理能力方面要远远优于MyISAM的表级锁定的。当系统并发量比较高的时候,InnoDB...
    99+
    2024-04-02
  • Java 缓存:如何优化系统性能?
    在开发 Java 应用程序时,我们经常需要处理大量数据。为了提高程序的性能,缓存是一个非常有用的技术。缓存可以将一些经常使用的数据存储在内存中,以避免频繁的数据库或网络请求,从而大大提高系统的响应速度和吞吐量。 本文将介绍 Java 缓存...
    99+
    2023-10-06
    缓存 学习笔记 面试
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作