Java synchronized锁升级jol过程详解

作者:Katsu 时间:2023-04-15 04:58:51 

jol(java object layout)需要的依赖


<dependency>
     <groupId>org.openjdk.jol</groupId>
     <artifactId>jol-core</artifactId>
     <version>0.10</version>
   </dependency>

一。synchronized锁对象的升级(膨胀)过程主要如下:

Java synchronized锁升级jol过程详解

1.膨胀过程:无锁(锁对象初始化时)-> 偏向锁(有线程请求锁) -> 轻量级锁(多线程轻度竞争)-> 重量级锁(线程过多或长耗时操作,线程自旋过度消耗cpu);

2.jvm默认延时4s自动开启偏向锁(此时为匿名偏向锁,不指向任务线程),可通过-XX:BiasedLockingStartUpDelay=0取消延时;如果不要偏向锁,可通过-XX:-UseBiasedLocking = false来设置

3.锁只能升级,不能降级;偏向锁可以被重置为无锁状态

4.锁对象头记录占用锁的线程信息,但不能主动释放,线程栈同时记录锁的使用信息,当有其他线程(T1)申请已经被占用的锁时,先根据锁对向的信息,找对应线程栈,若线程已结束,则锁对象先被置为无锁状态,再被T1线程占有后置为偏向锁;若线程位结束,则锁状态由当前偏向锁升级为轻量级锁。

5.偏向锁和轻量级锁在用户态维护,重量级锁需要切换到内核态(os)进行维护;

二。锁对象头(markword部分,8字节)使用不同的状态进行表示,64位虚拟机的markword如下所示:

Java synchronized锁升级jol过程详解

使用jol演示如下:

1.无锁状态

Object object = new Object(); System.out.println("hash: " + object.hashCode()); System.out.println(ClassLayout.parseInstance(object).toPrintable());

Java synchronized锁升级jol过程详解

header中前8个字节按照平时习惯的从高位到低位的展示为:00000000 00000000 00000000 00111001 10101110 11101101 00101111 00000001

对照上图,最后3位是001,无锁状态,中间31位(0111001 10101110 11101101 00101111)换算成十进制即为上图打印的hash:967765295

2.匿名偏向锁和偏向锁


Thread.sleep(5000); //等待jvm开启偏向锁
   Object o = new Object();
   System.out.println(ClassLayout.parseInstance(o).toPrintable());

synchronized (o){
     System.out.println(ClassLayout.parseInstance(o).toPrintable());
   }

Java synchronized锁升级jol过程详解

第一次打印为匿名偏向,第二次偏向锁指向了main线程

注意:用run启动程序,不要用debug,实验的时候,用debug启动,第二次打印直接升级轻量级锁。

3.轻量级锁


public static void main(String[] args) throws InterruptedException {
   Thread.sleep(5000);
   Object o = new Object();
   synchronized (o) {
     System.out.println(ClassLayout.parseInstance(o).toPrintable());
   }
   for (int i = 0; i < 1; i++) {
     Thread t = new Thread(() -> {
       print(o);
     });
     t.start();
   }
 }

public static void print(Object o) {
   synchronized (o){
     System.out.println(ClassLayout.parseInstance(o).toPrintable());
   }
 }

Java synchronized锁升级jol过程详解

4.重量级锁


public static void main(String[] args){
   Object o = new Object();
   for (int i = 0; i < 2; i++) {
     Thread t = new Thread(() -> {
       print(o);
     });
     t.start();
   }
 }

public static void print(Object o) {
   synchronized (o){
     System.out.println(ClassLayout.parseInstance(o).toPrintable());
   }
 }

Java synchronized锁升级jol过程详解

来源:https://www.cnblogs.com/katsu2017/p/12610002.html

标签:Java,synchronized,锁,升级,jol
0
投稿

猜你喜欢

  • Java数组与堆栈相关知识总结

    2023-11-12 06:12:18
  • ReadWriteLock接口及其实现ReentrantReadWriteLock方法

    2023-11-24 01:46:52
  • java8 实现提取集合对象的每个属性

    2023-10-17 19:37:27
  • 一篇带你解析入门LongAdder源码

    2023-11-28 20:17:52
  • spring webflux自定义netty 参数解析

    2023-07-26 18:38:25
  • 深入浅析Java反射机制

    2023-11-25 07:02:03
  • java 开发中网络编程之IP、URL详解及实例代码

    2023-08-06 10:26:29
  • IDEA中sout快捷键无效问题的解决方法

    2023-11-28 21:27:02
  • java实现单词搜索迷宫游戏

    2023-11-10 22:44:32
  • Java内存区域和内存模型讲解

    2023-11-26 12:08:39
  • Java深入学习图形用户界面GUI之事件处理

    2023-11-29 14:14:57
  • SpringBoot跨域Access-Control-Allow-Origin实现解析

    2023-11-28 23:04:34
  • Java实现配置加载机制

    2023-11-26 09:03:38
  • opencv3/C++图像滤波实现方式

    2023-06-23 15:37:08
  • springboot整合shiro与自定义过滤器的全过程

    2023-11-24 21:02:40
  • springAop实现权限管理数据校验操作日志的场景分析

    2023-11-23 10:00:27
  • spring boot整合log4j2及MQ消费处理系统日志示例

    2023-06-17 17:47:54
  • java 动态增加定时任务示例

    2023-07-29 06:56:00
  • JAVA如何按字节截取字符串

    2023-11-25 13:31:41
  • C语言时间函数之strftime()详解

    2023-06-26 02:42:32
  • asp之家 软件编程 m.aspxhome.com