一、对象的创建过程
1. class loading
2. class linking( verification , preparation , resolution )
3. class initiallizing
class字节码文件 load 进内存、检查格式、静态变量赋默认值、解析class字节码的变量指向常量池的指针为可以访问的物理地址,静态变量赋初始值
4. new 指令 申请对象内存,成员变量赋默认值
5. invockspection 指令
5.1 成员变量顺序赋初始值
5.2 执行构造方法语句
二、对象的存储布局
2.1 普通对象
1. 对象头 markword 8 字节
2. ClassPoint指针:类型指针 默认8 字节,可以-XX:+UseCompressedClassPointers为4字节,这个指针指向这个class文件
3. 实例数据
int 4 Byte
byte 1 Byte
引用类型8字节,开启-XX:+UserCompressedOops为4字节
4. Padding 对齐为8的倍数
2.2 数组对象
- 对象头:markword 8
- ClassPoint指针 8字节
- 数组长度 4字节
- 数组数据
- 对齐
三、对象头markword具体包括什么?8字节 64bit
3.1 锁信息
偏向锁,某个线程给占住了某个资源,下次这个线程再来的时候不用加锁
3.2 GC信息
分代年龄:被回收了多少次
为什么GC年龄默认最大15?
因为在markword中分代年龄占4bit,1111 即15
3.3 对象的hashcode
如果对象的hashcode被重写写死了hashcode,那这个hashcode就是这个值
没有被重写过的话,JVM会根据对象的那些后面的值算出来一个hashcode,即identitly hashcode
当一个对象计算过identityHashCode之后,就不能进入偏向锁状态了
即 偏向锁计算identity hashcode会进行锁撤销,膨胀为重量级锁
而重量级锁的ObjectMonitor类中有字段可以记录非加锁状态下的markword,可以存identity hashcode
四、对象定位 T t=new T()
4. 1 句柄池(间接指针)
指向两个指针,一个指向真正的对象,另一个指向T.class
这种对象定位在垃圾回收GC时效率较高。(三色标记法)
4.2 直接指针
指向真正的对象,这个对象有个指针指向T.class
五、对象怎么分配
new 出来一个对象,如果很小,就直接在栈上分配,pop弹出对象就没了。
如果很大,就直接分到堆内存老年代,
如果不大不小,现在线程本地进行分配。分配不下就放到伊甸园区,年龄到了就到老年代,年龄不到就在幸存区和伊甸园区不断gc