1、Object object = new Object()谈谈你对这句话的理解?

一般而言JDK8按照默认情况下,new一个对象占多少内存空间

位置所在:JVM里堆→新生区→伊甸园区

构成布局:头体?想想我们的HTML报文


2、对象在堆内存中布局

2.1、权威定义

周志明老师JVM第3版

java实现头部位置识别 java程序头部_数据


java实现头部位置识别 java程序头部_数据_02


2.2、对象在堆内存中的存储布局

2.2.1、对象头

对象标记Mark Word

它保存什么

java实现头部位置识别 java程序头部_java实现头部位置识别_03


java实现头部位置识别 java程序头部_java_04

在 64 位系统中, Mark Word 占了 8 个字节,类型指针占了 8 个字节,一共是 16 个字节 

java实现头部位置识别 java程序头部_sed_05

默认存储对象的HashCode、分代年龄和锁标志位等信息。

这些信息都是与对象自身定义无关的数据,所以MarkWord被设计成一个非固定的数据结构以便在极小的空间内存存储尽量多的数据。

它会根据对象的状态复用自己的存储空间,也就是说在运行期间MarkWord里存储的数据会随着锁标志位的变化而变化。


类元信息(又叫类型指针)

参考尚硅谷宋红康老师原图

java实现头部位置识别 java程序头部_sed_06

对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。


对象头多大

在64位系统中,Mark Word占了8个字节,类型指针占了8个字节,一共是16个字节。


2.2.2、实例数据

存放类的属性(Field)数据信息,包括父类的属性信息,如果是数组的实例部分还包括数组的长度,这部分内存按4字节对齐。


2.2.3、对齐填充

虚拟机要求对象起始地址必须是8字节的整数倍。填充数据不是必须存在的,仅仅是为了字节对齐这部分内存按8字节补充对齐。


2.3、官网理论

Hotspot术语表官网 HotSpot Glossary of Terms

java实现头部位置识别 java程序头部_sed_07

底层源码理论证明 jdk8u/jdk8u/hotspot: 89fb452b3688 src/share/vm/oops/oop.hpp

_mark字段是mark word,_metadata是类指针klass pointer, 

对象头(object header)即是由这两个字段组成,这些术语可以参考Hotspot术语表,


java实现头部位置识别 java程序头部_sed_08


3、再说对象头的MarkWord

3.1、32位(看一下即可,不用学了,以64位为准)

java实现头部位置识别 java程序头部_juc_09


3.2、64位重要

oop.hpp

java实现头部位置识别 java程序头部_juc_10


markOop.hpp

hash: 保存对象的哈希码

age: 保存对象的分代年龄 

biased_lock: 偏向锁标识位 

lock: 锁状态标识位 

JavaThread* :保存持有偏向锁的线程ID 

epoch: 保存偏向时间戳 

java实现头部位置识别 java程序头部_java_11


markword(64位)分布图,

对象布局、GC回收和后面的锁升级就是对象标记MarkWord里面标志位的变化

java实现头部位置识别 java程序头部_数据_12


4、聊聊Object obj = new Object()

JOL证明

JOL官网:OpenJDK: jol

POM


<!-- 定位:分析对象在JVM的大小和分布 -->
<dependency>
    <groupId>org.openjdk.jol</groupId>
    <artifactId>jol-core</artifactId>
    <version>0.9</version>
    <scope>provided</scope>
</dependency>

小试一下


package com.atguigu.juc.test;

import org.openjdk.jol.info.ClassLayout;
import org.openjdk.jol.vm.VM;

/**
 * @author shizan
 * @Classname MyObject
 * @Description TODO
 * @Date 2022/6/7 8:40 下午
 */
public class MyObject {
    public static void main(String[] args) {
        //VM 的细节详细情况
        System.out.println(VM.current().details());
        // 所有的对象分配的字节都是 8 的整数倍。
        System.out.println(VM.current().objectAlignment());
    }
}


java实现头部位置识别 java程序头部_juc_13



代码

package com.atguigu.juc.test;


import org.openjdk.jol.info.ClassLayout;


/**

 * @author shizan

 * @Classname JOLDemo

 * @Description TODO

 * @Date 2022/6/8 4:16 下午

 */

public class JOLDemo {

    public static void main(String[] args) {

        Object o = new Object();

        System.out.println(ClassLayout.parseInstance(o).toPrintable());

    }

}

结果呈现说明

java实现头部位置识别 java程序头部_数据_14



OFFSET


偏移量,也就是到这个字段位置所占用的byte数 


SIZE


后面类型的字节大小 


TYPE


是Class中定义的类型 


DESCRIPTION


DESCRIPTION是类型的描述 


VALUE


VALUE是TYPE在内存中的值 


GC年龄采用4位bit存储,最大为15,

例如MaxTenuringThreshold参数默认值就是15



尾巴参数说明

命令 java -XX:+PrintCommandLineFlags -version

默认开启压缩说明

java实现头部位置识别 java程序头部_sed_15

-XX:+UseCompressedClassPointers

结果

java实现头部位置识别 java程序头部_java实现头部位置识别_16

上述表示开启了类型指针的压缩,以节约空间,假如不加压缩???


手动关闭压缩再看看

-XX:-UseCompressedClassPointers

结果

java实现头部位置识别 java程序头部_sed_17


5、换成其他对象试试

结果

java实现头部位置识别 java程序头部_juc_18

java实现头部位置识别 java程序头部_sed_19