目录
补充1(对象的内存布局):一个Java对象,在堆中都需要存储它的什么信息?
补充2(对象的2种访问定位方式):我们知道,虚拟机栈中存储了对象的引用(就是一个地址吧,我觉得),那怎么通过这个引用来获得这个对象的所有信息?
一、进行类加载
二、堆中分配内存
1、怎么输出GC日志:
--XX:+PringGCDetails
2、内存分配的两种方式:
- 如果JVM带有空间压缩功能,那堆空间就比较规整,这时候可以用指针碰撞(空闲区和已使用区之间的指针向空闲区移动);
- 反之,用空闲列表,典型如使用标记-清除算法的CMS;
3、内存分配过程中并发控制的两种方式:
- 方案一:CAS乐观锁;
- 方案二:给每一个线程单独开一个内存缓冲区,每个线程的缓冲区用完后采用CAS;
三、内存空间初始化
初始化为零值;
四、对象头初始化(对象头包含哪些信息?)
① 对象的哈希码:哈希码和对象存储地址有关,可以通过obj.toString()来看哈希码是啥,toString()函数实现如下:
public String toString() { return getClass().getName() + "@" + Integer.toHexString(hashCode()); }
② GC分代年龄:
③ 一些锁信息:
五、执行构造函数。
补充1(对象的内存布局):一个Java对象,在堆中都需要存储它的什么信息?
- 对象头:哈希码、gc年龄、锁信息等等;
- 实例数据部分:用户代码定义的内容;
- 对齐填充:补位用;
补充2(对象的2种访问定位方式):我们知道,虚拟机栈中存储了对象的引用(就是一个地址吧,我觉得),那怎么通过这个引用来获得这个对象的所有信息?
- 直接指针:引用存储的就是对象在Java堆中的地址,在《深入理解JVM》这本书中,对象实例存储在堆里,但对象类型信息存储在方法区,所以堆中还需要存储一个方法区的地址;
- 句柄:在堆中开一个句柄池,分别存储对象实例的地址和对象类型信息的地址;
- 对比:对于直接指针,因为少了一层访问,所以相比句柄方法速度快,但当对象位置移动频繁时,栈中的地址就得经常变化。而句柄这种方式,在对象位置移动时,只需要修改句柄池的信息。
- 那为什么修改句柄池就比修改虚拟机栈要优呢?
- 书中的图只画了对象实例对应栈中一个引用的情况,实际上因为虚拟机栈中会有多个引用同时指向堆中的对象实例,所以对象实例位置改变,这些引用都要改变,而句柄就相当于对这些引用进行了一次汇总。