以下面代码为例,来说明对象定位的过程:
class Bus extends Car { private String code; private String color; Bus(String code, String color) { this.code = code; this.color = color; } // 省略其他方法... } public class ReferenceTest { Bus myBus = new Bus("Java中文社群", "蓝色"); } 复制代码
以官方默认的 HotSpot 虚拟机来说, myBus
就是存储在本地变量表中 reference 类型的变量, new Bus("Java中文社群", "蓝色")
就是存储在 Java 堆中的对象实例数据,它存储了此实体类的所有字段信息,例如 code="Java中文社群"
以及 color="蓝色"
等信息,而 Java 堆中的还存储着对象类型数据的地址,它存储的是对象的类型信息,还有它的父类信息等。
由于 reference 类型在《Java虚拟机规范》里面只规定了它是一个指向对象的引用,并没有定义这个引用应该通过什么方式去定位、访问到堆中对象的具体位置,所以对象访问方式也是由虚拟机实现而定的,主流的访问方式主要有使用句柄和直接指针两种:
因此使用句柄来访问的最大好处就是 reference 中存储的是稳定句柄地址,在对象被移动(垃圾收集时移动对象是非常普遍的行为)时只会改变句柄中的实例数据指针,而 reference 本身不需要被修改。使用直接指针访问速度更快,但如果对象被移动则需要修改 reference 本身。
由于对象访问在 Java 中非常频繁,因此这类开销积少成多也是一项极为可观的执行成本,所以官方默认的 HotSpot 虚拟机采用的就是「 直接指针 」来定位对象的。