对象的内存布局和访问定位
对象的内存布局
对象的内存布局分为3个部分,分别是对象头,实例数据,对其填充,下面分别介绍一下:
- 对象头:对象头包含两部分信息,分别是对象自身的运行时数据和类型指针
- 运行时数据:包含对象的哈希码,GC分代年龄,锁状态标志,线程持有的锁,偏向线程ID,偏向时间戳等
- 类型指针:是指向对象的类元数据的指针,用于确定该对象是哪个类的实例
- 实例数据:对象真正存储的有效信息,也就是对象的各个字段的内容,包括父类的字段内容。
- 对齐填充:HotSpot VM虚拟机的自动内存管理系统要求对象的内存必须是8bit的倍数,对象头部分正好是8bit字节的倍数,但是运行时数据不一定,当运行时数据不满足这个要求时,用对齐填充来满足这个条件,由此可以看出,这个并不是必须的。
对象的访问定位
对象创建之后,我们需要使用栈中的reference类型去引用对象,java规范中只规定了一个指向对象的引用,并没有规定如何去引用,因此具体的引用方式跟虚拟机不同而不一样,目前的引用方式有两种,分别是句柄和直接指针。
- 句柄:在java堆中划分出一个内存出来,用来做句柄池,句柄中包含了对象的实例数据和类型数据各自的地址信息,在栈中的reference保存的则是句柄的地址。使用句柄的优点是,当对象被移动时,reference中的值不用改变,只需要改变句柄中的值就可以了;缺点是对了一次查找的开销
- 直接指针:栈中的renference类型存储的就是对象的直接地址,而对象的实例中保存了该对象类型数据地址的指针,使用直接指针的优点是,速度更快,节省了一次查找的开销。(HotSpot使用的就是这种方式)