JVM:对象的实例化、内存布局与访问定位
大厂面试题:
美团:
对象在JVM中是怎么存储的?
对象头信息里面有哪些东西?
蚂蚁金服:
Java对象头有什么?
1.对象的实例化方式:
- new
- Class的newInstance
- Constractor的newInstance
- clone()
- 使用反序列化:从文件、网络中读取一个对象的二进制流
- 第三方库
2.对象的创建过程
从字节码角度
0: new 在堆中创建
3: dup 在栈中引用
4: invoke special 调用构造器
3.对象的内存布局
对象头(Header)
实例数据
对齐填充
3.对象的内存布局
/**
* 测试对象实例化的过程
* ① 加载类元信息 - ② 为对象分配内存 - ③ 处理并发问题 - ④ 属性的默认初始化(零值初始化)
* - ⑤ 设置对象头的信息 - ⑥ 属性的显式初始化、代码块中初始化、构造器中初始化
*
*
* 给对象的属性赋值的操作:
* ① 属性的默认初始化 - ② 显式初始化 / ③ 代码块中初始化 - ④ 构造器中初始化
*/
public class Customer{
int id = 1001;
String name;
Account acct;
{
name = "匿名客户";
}
public Customer(){
acct = new Account();
}
}
class Account{
}
对象头
对象头包含了两部分,分别是运行时元数据(Mark Word)和类型指针。如果是数组,还需要记录数组的长度
运行时元数据
哈希值(HashCode)
GC分代年龄
锁状态标志
线程持有的锁
偏向线程ID
偏向时间戳
类型指针
- 指向类元数据InstanceKlass,确定该对象所属的类型。
实例数据
它是对象真正存储的有效信息,包括程序代码中定义的各种类型的字段(包括从父类继承下来的和本身拥有的字段) - 相同宽度的字段总是被分配在一起
- 父类中定义的变量会出现在子类之前
- 如果CompactFields参数为true(默认为true):子类的窄变量可能插入到父类变量的空隙
对齐填充
不是必须的,也没有特别的含义,仅仅起到占位符的作用
4.对象的访问定位
JVM是如何通过栈帧中的对象引用访问到其内部的对象实例呢?
句柄访问
直接指针(HotSpot采用)
直接指针是局部变量表中的引用,直接指向堆中的实例,在对象实例中有类型指针,指向的是方法区中的对象类型数据