对象访问
在Java中即使是最简单的对象访问,也会涉及到栈、堆、方法区这三个最重要的内存区域之间的关系,如:
Object obj = new Object();
假设这句代码出现在方法体中,那 “Object obj” 这部分的语义将会反映到Java栈的本地变量表中,作为reference类型数据出现。而 “new Object()” 这部分的语义将会反映到Java堆中,形成一块存储了Object类型所有实例数据值(Instance Data,对象中各个实例字段的数据)的结构化内存,根据具体类型以及虚拟机实现的对象内存布局的不同,这块内存的长度是不固定的。另外,在Java堆中还必须包含能查找到此对象类型数据(如对象类型、父类、实现的接口、方法等)的地址信息,这些类型数据则存储在方法区中。
不同的虚拟机实现的对象访问方式会有所不同,主流的访问方式有两种:使用句柄和直接指针。
使用句柄访问方式:Java堆中将会划分出一块内存来作为句柄池,reference中存储的就是对象的句柄地址,而句柄中包含了对象实例数据和类型数据各自的具体地址信息,如图:
使用直接指针访问方式:Java堆对象的布局必须考虑如何放置访问类型数据的相关信息,reference中直接存储的是对象地址。
这两种对象的访问方式各有优势:
使用句柄访问方式的最大好处就是reference中存储的是稳定的句柄地址,在对象被移动(垃圾收集时移动对象是非常普遍的行为)时只会改变句柄中的实例数据指针,而feference本身不需要改变。
使用直接指针访问的方式的最大好处就是速度更快,它节省了一次指针定位的时间开销。
Sun HotSpot虚拟机使用的是直接指针访问方式。