使用jol查看synchronized锁信息

简介: 使用jol查看synchronized锁信息

JOL

JOL 是OpenJdk发布的用查看对象在JVM中信息,全称是Java Object Layout。可以通过命令行使用。
下面例子中仅使用到了ClassLayout,有兴趣可以看看官网介绍。

pom依赖,jol 0.16版本没有使用字节码展示对象头而是直接展示了锁信息,与jstack日志一致,如果想查看对象头变化请使用 0.10版

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

synchronization

synchronization是同步关键字,下面的示例使用synchronization+jol查看锁信息。有兴趣可以直接运行看看效果,也可以直接拖到最后看看结论或者看看oracle官方解释。

JAVA代码

import lombok.extern.slf4j.Slf4j;
import org.openjdk.jol.info.ClassLayout;

@Slf4j
public class SynchronizedDemo {

    public static void main(String[] args) {
        SynchronizedDemo demo = new SynchronizedDemo();
        SynchronizedDemo demo1 = new SynchronizedDemo();
        SynchronizedDemo demo2 = new SynchronizedDemo();
        new Thread(() -> {
            demo.lockMethod();
        }).start();
        new Thread(() -> {
            demo.lockInstance(demo1);
        }).start();
        new Thread(() -> {
            SynchronizedDemo.lockStaticMethod();
        }).start();
        log.info("打印demo头信息");
        log.info(ClassLayout.parseInstance(demo).toPrintable());
        log.info("打印demo1头信息");
        log.info(ClassLayout.parseInstance(demo1).toPrintable());
        log.info("打印demo2头信息");
        log.info(ClassLayout.parseInstance(demo2).toPrintable());
        log.info("打印Class头信息");
        log.info(ClassLayout.parseInstance(SynchronizedDemo.class).toPrintable());
    }

    public synchronized static void lockStaticMethod() {
        try {
            Thread.sleep(1000000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    public synchronized void lockMethod() {
        try {
            Thread.sleep(1000000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    public void lockInstance(SynchronizedDemo demo) {
        synchronized (demo) {
            try {
                Thread.sleep(1000000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

打印日志

2022-06-15 15:28:48 INFO    --- [main      ] c.m.basis.concurrent.SynchronizedDemo    : 打印demo头信息
2022-06-15 15:28:49 INFO    --- [main      ] c.m.basis.concurrent.SynchronizedDemo    : com.meijm.basis.concurrent.SynchronizedDemo object internals:
OFF  SZ   TYPE DESCRIPTION               VALUE
  0   8        (object header: mark)     0x000000002097f050 (thin lock: 0x000000002097f050)
  8   4        (object header: class)    0xf800c105
 12   4        (object alignment gap)    
Instance size: 16 bytes
Space losses: 0 bytes internal + 4 bytes external = 4 bytes total

2022-06-15 15:28:49 INFO    --- [main      ] c.m.basis.concurrent.SynchronizedDemo    : 打印demo1头信息
2022-06-15 15:28:49 INFO    --- [main      ] c.m.basis.concurrent.SynchronizedDemo    : com.meijm.basis.concurrent.SynchronizedDemo object internals:
OFF  SZ   TYPE DESCRIPTION               VALUE
  0   8        (object header: mark)     0x0000000020a7ee10 (thin lock: 0x0000000020a7ee10)
  8   4        (object header: class)    0xf800c105
 12   4        (object alignment gap)    
Instance size: 16 bytes
Space losses: 0 bytes internal + 4 bytes external = 4 bytes total

2022-06-15 15:28:49 INFO    --- [main      ] c.m.basis.concurrent.SynchronizedDemo    : 打印demo2头信息
2022-06-15 15:28:49 INFO    --- [main      ] c.m.basis.concurrent.SynchronizedDemo    : com.meijm.basis.concurrent.SynchronizedDemo object internals:
OFF  SZ   TYPE DESCRIPTION               VALUE
  0   8        (object header: mark)     0x0000000000000001 (non-biasable; age: 0)
  8   4        (object header: class)    0xf800c105
 12   4        (object alignment gap)    
Instance size: 16 bytes
Space losses: 0 bytes internal + 4 bytes external = 4 bytes total

2022-06-15 15:28:49 INFO    --- [main      ] c.m.basis.concurrent.SynchronizedDemo    : 打印Class头信息
2022-06-15 15:28:49 INFO    --- [main      ] c.m.basis.concurrent.SynchronizedDemo    : java.lang.Class object internals:
OFF  SZ                                              TYPE DESCRIPTION                    VALUE
  0   8                                                   (object header: mark)          0x000000001ba5570a (fat lock: 0x000000001ba5570a)
  8   4                                                   (object header: class)         0xf80003df
 12   4                     java.lang.reflect.Constructor Class.cachedConstructor        null
 16   4                                   java.lang.Class Class.newInstanceCallerCache   null
 20   4                                  java.lang.String Class.name                     (object)
 24   4                                                   (alignment/padding gap)        
 28   4                       java.lang.ref.SoftReference Class.reflectionData           (object)
 32   4   sun.reflect.generics.repository.ClassRepository Class.genericInfo              null
 36   4                                java.lang.Object[] Class.enumConstants            null
 40   4                                     java.util.Map Class.enumConstantDirectory    null
 44   4                    java.lang.Class.AnnotationData Class.annotationData           (object)
 48   4             sun.reflect.annotation.AnnotationType Class.annotationType           null
 52   4                java.lang.ClassValue.ClassValueMap Class.classValueMap            null
 56  32                                                   (alignment/padding gap)        
 88   4                                               int Class.classRedefinedCount      0
 92   4                                                   (object alignment gap)         
Instance size: 512 bytes
Space losses: 36 bytes internal + 4 bytes external = 40 bytes total
在代码运行时可使用jstack -l 进程id,查看具体锁信息,

结论

synchronization锁定的是JVM中运行的对象,静态方法锁定的是Class对象,代码块锁定的是括号中的对象。

参考资料

https://github.com/openjdk/jol

https://docs.oracle.com/javase/tutorial/essential/concurrency/locksync.html

目录
相关文章
|
8月前
|
Oracle JavaScript Java
JDK的版本迭代特性(JDK9 - JDK20)
JDK的版本迭代特性(JDK9 - JDK20)
|
缓存
IDEA找不到或无法加载主类
IDEA找不到或无法加载主类
3124 0
IDEA找不到或无法加载主类
|
4月前
|
Oracle Java iOS开发
JDK的选型、安装与配置
JDK的选型、安装与配置
141 1
|
4月前
|
Oracle Java iOS开发
JDK的选型、安装与配置
JDK的选型、安装与配置
141 1
|
5月前
|
前端开发 关系型数据库 MySQL
com.mysql.jdbc.Driver 和 com.mysql.cj.jdbc.Driver 的区别
这篇文章讨论了`com.mysql.jdbc.Driver`和`com.mysql.cj.jdbc.Driver`两个MySQL驱动类的区别,指出`com.mysql.jdbc.Driver`适用于MySQL 5的`mysql-connector-java`版本,而`com.mysql.cj.jdbc.Driver`适用于MySQL 6及以上版本的`mysql-connector-java`。文章还提到了在实际使用中如何根据MySQL版本选择合适的驱动类。
com.mysql.jdbc.Driver 和 com.mysql.cj.jdbc.Driver 的区别
|
6月前
|
消息中间件 前端开发 Android开发
Android面试题自定义View之Window、ViewRootImpl和View的三大流程
Android开发中,View的三大核心流程包括measure(测量)、layout(布局)和draw(绘制)。MeasureSpec类在测量过程中起到关键作用,它结合尺寸大小和模式(EXACTLY、AT_MOST、UNSPECIFIED)来指定View应如何测量。onMeasure方法用于自定义View的测量,布局阶段,ViewGroup调用onLayout确定子元素位置,而draw阶段按照特定顺序绘制背景、内容、子元素和装饰。整个流程始于ViewRootImpl的performTraversals,该方法触发测量、布局和绘制。
124 0
|
7月前
|
Java
JAVA实现CRC16多项式(GB/T20999-2017)
JAVA实现CRC16多项式(GB/T20999-2017)
158 0
|
存储 关系型数据库 MySQL
从一个案例深入剖析InnoDB隐式锁和可见性判断(4)
从一个案例深入剖析InnoDB隐式锁和可见性判断
698 0
|
SQL 关系型数据库 MySQL
从一个案例深入剖析InnoDB隐式锁和可见性判断(2)
从一个案例深入剖析InnoDB隐式锁和可见性判断
149 0
从一个案例深入剖析InnoDB隐式锁和可见性判断(2)
|
NoSQL 关系型数据库 索引
从一个案例深入剖析InnoDB隐式锁和可见性判断(1)
从一个案例深入剖析InnoDB隐式锁和可见性判断
155 0
从一个案例深入剖析InnoDB隐式锁和可见性判断(1)