java通过jol-core库分析对象内存分布以及查看Synchronized锁升级过程

在Java开发中,我们可以借助jol-core这个库来查看对象内存信息。

引入依赖:

html 复制代码
<dependency>
      <groupId>org.openjdk.jol</groupId>
      <artifactId>jol-core</artifactId>
      <version>0.17</version>
</dependency>

要查看对象内存信息,我们只需要调用ClassLayout.parseInstance(object).toPrintable()

如下所示的代码,是查看两个对象信息:

java 复制代码
package com.xxx.mem;
import org.openjdk.jol.info.ClassLayout;
public class ObjectPrintExample {
    public static void main(String[] args) {
        Object object = new Object();
        System.out.println(ClassLayout.parseInstance(object).toPrintable());
        Object[] arr = new Object[10];
        System.out.println(ClassLayout.parseInstance(arr).toPrintable());
    }
}

运行程序,打印信息如下所示:

从打印内容来看,我们看到了对象包含:对象头,对象体,对齐字节。

其中对象头包括: mark word,klass word,数组长度。如果对象不是数组,数组长度可以忽略不记,上图中,我们打印object,数组长度就没有体现(array length)。

我们这里重点关注一下对象头中的mark word,如下所示:

这里面通过对象头64位中的最后3位的值,我们看到它们取值不同,可能会出现几种状态,其中无锁 、偏向锁、轻量级锁,重量级锁正是Sychronized锁升级过程中表现出来的升级状态。

下面我们通过一段代码的运行来感受一下这几个状态的变化:

java 复制代码
package com.xxx.mem;
import org.openjdk.jol.info.ClassLayout;
import java.util.concurrent.TimeUnit;
public class LayoutPrintExample {
    public static void main(String[] args) throws InterruptedException {
        Object o = new Object();
        // Step 1
        System.out.println(ClassLayout.parseInstance(new Object()).toPrintable());
        TimeUnit.SECONDS.sleep(4);
        // Step 2
        System.out.println(ClassLayout.parseInstance(new Object()).toPrintable());
        //Step 3
        synchronized(o) {
            System.out.println(ClassLayout.parseInstance(o).toPrintable());
        }
        // Step 4
        new Thread(()-> {
            synchronized(o) {
                System.out.println(ClassLayout.parseInstance(o).toPrintable());
            }
        }).start();
        // Step 5
        new Thread(()-> {
            synchronized(o) {
                System.out.println(ClassLayout.parseInstance(o).toPrintable());
            }
        }).start();
    }
}

这段代码直接运行,可能达不到我们需要的效果,因为jdk默认不会开启偏向锁,所以我们要手动开启,并且设置一个延时4秒,这样我们正好看清整个锁升级过程。

虚拟机参数:-XX:BiasedLockingStartupDelay=4 -XX:+UseBiasedLocking

运行程序,打印结果:

这个结果的解释:

step1,程序刚开始运行,因为开启了偏向锁,第一次打印无锁状态。

step2,这里睡眠4秒,相当于延时4秒,虚拟机偏向锁生效,打印出偏向锁状态。

step3,由于程序调用synchronized,对象锁升级为轻量级锁。

step4,在线程中调用synchronized,这时候竞争并不是很激烈,还是轻量级锁。

step5,在另一个线程中调用synchronized,相当于多线程环境,这时候竞争激烈,自身cas已经无法满足需求,对象锁升级为重量级锁。

这里我们通过对象头markword也看出了锁状态,分别是non-biasable,biasable,thin lock,thin lock,fat lock。另外,通过16进制对应的二进制结果也验证了一下markword对应后三位状态。

jol-core可能因为版本不同,显示的markword,可能是十六进制,也有可能是二进制,如果是二进制,我们需要注意,是否为小端序、大端序,如果端序搞错,结果有可能不是我们期望的锁状态。

相关推荐
_日拱一卒4 分钟前
LeetCode:矩阵置零
java·数据结构·线性代数·算法·leetcode·职场和发展·矩阵
weixin_408099678 分钟前
【实战教程】懒人精灵如何实现 OCR 文字识别?接口调用完整指南(附可运行示例)
java·前端·人工智能·后端·ocr·api·懒人精灵
花千树-01025 分钟前
Java Agent 集成 MCP 工具协议:让 AI 真正驱动企业系统
java·ai·langchain·ai agent·mcp·harness·j-langchain
橘子编程30 分钟前
GoF 23 种设计模式完整知识总结与使用教程
java·c语言·开发语言·python·设计模式
君以思为故35 分钟前
认识Linux -- 线程同步与互斥
java·开发语言
被摘下的星星40 分钟前
Java接口需要注意的细节
java·开发语言
培风图南以星河揽胜43 分钟前
幻想简历!博主本人期望的 AI Agent 全栈简历:Java + Python + Vue3 跨语言实战,代码已开源!
java·人工智能·python
鬼先生_sir1 小时前
Dubbo:从入门到精通
java·dubbo·springcloud
艾莉丝努力练剑1 小时前
【Linux线程】Linux系统多线程(四):线程ID及进程地址空间布局,线程封装
java·linux·运维·服务器·c语言·c++·学习
有味道的男人1 小时前
电商效率翻倍:用 Open Claw 对接 1688 接口,快速实现图片选品 + 货源监控
java·开发语言·数据库