java 偏向锁 重偏向_盘一盘 synchronized (二)—— 偏向锁批量重偏向与批量撤销...

2023-10-21 23:30

本文主要是介绍java 偏向锁 重偏向_盘一盘 synchronized (二)—— 偏向锁批量重偏向与批量撤销...,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在本文讲解之前,先来简单了解一下为什么会有批量重偏向和批量撤销。   批量重偏向:当一个线程创建了大量对象并执行了初始的同步操作,后来另一个线程也来将这些对象作为锁对象进行操作,会导偏向锁重偏向的操作。 批量撤销:在多线程竞争剧烈的情况下,使用偏向锁将会降低效率,于是乎产生了批量撤销机制。

JVM的默认参数值

通过JVM的默认参数值,找一找批量重偏向和批量撤销的阈值。 设置JVM参数-XX:+PrintFlagsFinal,在项目启动时即可输出JVM的默认参数值

36e16348f951c4138457355d43293c51.png   intx BiasedLockingBulkRebiasThreshold   = 20   默认偏向锁批量重偏向阈值 intx BiasedLockingBulkRevokeThreshold  = 40   默认偏向锁批量撤销阈值 当然我们可以通过-XX:BiasedLockingBulkRebiasThreshold 和 -XX:BiasedLockingBulkRevokeThreshold 来手动设置阈值

be2b189c439b0190089ef30f4e7e3bde.png

批量重偏向

public static void main(String[] args) throwsException {

//延时产生可偏向对象

Thread.sleep(5000); //创造100个偏向线程t1的偏向锁 List listA = new ArrayList<>(); Thread t1 = new Thread(() ->{ for (int i = 0; i <100 ; i++) { A a = newA(); synchronized(a){ listA.add(a); } } try{ //为了防止JVM线程复用,在创建完对象后,保持线程t1状态为存活 Thread.sleep(100000000); } catch(InterruptedException e) { e.printStackTrace(); } }); t1.start(); //睡眠3s钟保证线程t1创建对象完成 Thread.sleep(3000); out.println(\"打印t1线程,list中第20个对象的对象头:\"); out.println((ClassLayout.parseInstance(listA.get(19)).toPrintable())); //创建线程t2竞争线程t1中已经退出同步块的锁 Thread t2 = new Thread(() ->{ //这里面只循环了30次!!! for (int i = 0; i < 30; i++) { A a =listA.get(i); synchronized(a){ //分别打印第19次和第20次偏向锁重偏向结果 if(i==18||i==19){ out.println(\"第\"+ ( i + 1) + \"次偏向结果\"); out.println((ClassLayout.parseInstance(a).toPrintable())); } } } try{ Thread.sleep(10000000); } catch(InterruptedException e) { e.printStackTrace(); } }); t2.start(); Thread.sleep(3000); out.println(\"打印list中第11个对象的对象头:\"); out.println((ClassLayout.parseInstance(listA.get(10)).toPrintable())); out.println(\"打印list中第26个对象的对象头:\"); out.println((ClassLayout.parseInstance(listA.get(25)).toPrintable())); out.println(\"打印list中第41个对象的对象头:\"); out.println((ClassLayout.parseInstance(listA.get(40)).toPrintable())); }

我们一步一步来分析打印结果 首先,创造了100个偏向线程t1的偏向锁,结果没什么好说的,100个偏向锁嘛,偏向的线程ID信息为531939333

2821d7bb7ce5ddfc2528e9ceda3620bc.png

再来看看重偏向的结果,线程t2,前19次偏向均产生了轻量锁,而到第20次的时候,达到了批量重偏向的阈值20,此时锁并不是轻量级锁,而变成了偏向锁,此时偏向的线程t2,线程t2的ID信息为5322162821

04dbfbdb91fa5873ef257b03fd65c7bd.png

最后我们再来看一下偏向结束后的对象头信息。 前20个对象,并没有触发了批量重偏向机制,线程t2执行释放同步锁后,转变为无锁形态 第20~30个对象,触发了批量重偏向机制,对象为偏向锁状态,偏向线程t2,线程t2的ID信息为5322162821 而31个对象之后,也没有触发了批量重偏向机制,对象仍偏向线程t1,线程t1的ID信息为531939333

2dd7c08e97a1e47904da79d920cad7ef.png

批量撤销

我们再来看看批量撤销

public static void main(String[] args) throwsException {

Thread.sleep(5000); List listA = new ArrayList<>(); Thread t1 = new Thread(() ->{ for (int i = 0; i <100 ; i++) { A a = newA(); synchronized(a){ listA.add(a); } } try{ Thread.sleep(100000000); } catch(InterruptedException e) { e.printStackTrace(); } }); t1.start(); Thread.sleep(3000); Thread t2 = new Thread(() ->{ //这里循环了40次。达到了批量撤销的阈值 for (int i = 0; i < 40; i++) { A a =listA.get(i); synchronized(a){ } } try{ Thread.sleep(10000000); } catch(InterruptedException e) { e.printStackTrace(); } }); t2.start(); //———————————分割线,前面代码不再赘述—————————————————————————————————————————— Thread.sleep(3000); out.println(\"打印list中第11个对象的对象头:\"); out.println((ClassLayout.parseInstance(listA.get(10)).toPrintable())); out.println(\"打印list中第26个对象的对象头:\"); out.println((ClassLayout.parseInstance(listA.get(25)).toPrintable())); out.println(\"打印list中第90个对象的对象头:\"); out.println((ClassLayout.parseInstance(listA.get(89)).toPrintable())); Thread t3 = new Thread(() ->{ for (int i = 20; i < 40; i++) { A a =listA.get(i); synchronized(a){ if(i==20||i==22){ out.println(\"thread3 第\"+ i + \"次\"); out.println((ClassLayout.parseInstance(a).toPrintable())); } } } }); t3.start(); Thread.sleep(10000); out.println(\"重新输出新实例A\"); out.println((ClassLayout.parseInstance(newA()).toPrintable())); }

来看看输出结果,这部分和上面批量偏向结果的大相径庭。重点关注记录的线程ID信息 前20个对象,并没有触发了批量重偏向机制,线程t2执行释放同步锁后,转变为无锁形态 第20~40个对象,触发了批量重偏向机制,对象为偏向锁状态,偏向线程t2,线程t2的ID信息为540039429 而41个对象之后,也没有触发了批量重偏向机制,对象仍偏向线程t1,线程t1的ID信息为540002309

ea16a94575784341103d7555c3751b3b.png

再来看看最后新生成的对象A。值得注意的是:本应该为可偏向状态偏向锁的新对象,在经历过批量重偏向和批量撤销后直接在实例化后转为无锁。

cdbedb3a62369bd224c1c959b0cdf7a6.png

简单总结

1、批量重偏向和批量撤销是针对类的优化,和对象无关。 2、偏向锁重偏向一次之后不可再次重偏向。 3、当某个类已经触发批量撤销机制后,JVM会默认当前类产生了严重的问题,剥夺了该类的新实例对象使用偏向

这篇关于java 偏向锁 重偏向_盘一盘 synchronized (二)—— 偏向锁批量重偏向与批量撤销...的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/257642

相关文章

Python实现批量CSV转Excel的高性能处理方案

《Python实现批量CSV转Excel的高性能处理方案》在日常办公中,我们经常需要将CSV格式的数据转换为Excel文件,本文将介绍一个基于Python的高性能解决方案,感兴趣的小伙伴可以跟随小编一... 目录一、场景需求二、技术方案三、核心代码四、批量处理方案五、性能优化六、使用示例完整代码七、小结一、

Java实现将HTML文件与字符串转换为图片

《Java实现将HTML文件与字符串转换为图片》在Java开发中,我们经常会遇到将HTML内容转换为图片的需求,本文小编就来和大家详细讲讲如何使用FreeSpire.DocforJava库来实现这一功... 目录前言核心实现:html 转图片完整代码场景 1:转换本地 HTML 文件为图片场景 2:转换 H

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

C#实现一键批量合并PDF文档

《C#实现一键批量合并PDF文档》这篇文章主要为大家详细介绍了如何使用C#实现一键批量合并PDF文档功能,文中的示例代码简洁易懂,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言效果展示功能实现1、添加文件2、文件分组(书签)3、定义页码范围4、自定义显示5、定义页面尺寸6、PDF批量合并7、其他方法

SpringBoot实现不同接口指定上传文件大小的具体步骤

《SpringBoot实现不同接口指定上传文件大小的具体步骤》:本文主要介绍在SpringBoot中通过自定义注解、AOP拦截和配置文件实现不同接口上传文件大小限制的方法,强调需设置全局阈值远大于... 目录一  springboot实现不同接口指定文件大小1.1 思路说明1.2 工程启动说明二 具体实施2

Java实现在Word文档中添加文本水印和图片水印的操作指南

《Java实现在Word文档中添加文本水印和图片水印的操作指南》在当今数字时代,文档的自动化处理与安全防护变得尤为重要,无论是为了保护版权、推广品牌,还是为了在文档中加入特定的标识,为Word文档添加... 目录引言Spire.Doc for Java:高效Word文档处理的利器代码实战:使用Java为Wo

SpringBoot日志级别与日志分组详解

《SpringBoot日志级别与日志分组详解》文章介绍了日志级别(ALL至OFF)及其作用,说明SpringBoot默认日志级别为INFO,可通过application.properties调整全局或... 目录日志级别1、级别内容2、调整日志级别调整默认日志级别调整指定类的日志级别项目开发过程中,利用日志

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版

Spring 中的切面与事务结合使用完整示例

《Spring中的切面与事务结合使用完整示例》本文给大家介绍Spring中的切面与事务结合使用完整示例,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录 一、前置知识:Spring AOP 与 事务的关系 事务本质上就是一个“切面”二、核心组件三、完