在 Java 编程里,对象的创建和销毁是基础且关键的操作,深刻理解这一过程有助于编写出高效、稳定的代码。下面将详细阐述 Java 中对象的创建和销毁过程。
当代码中使用 new
关键字创建对象时,Java 虚拟机(JVM)首先会检查该对象对应的类是否已经被加载到内存中。如果尚未加载,JVM 会通过类加载器将该类的字节码文件加载到内存,并对其进行验证、准备和解析等操作,最终完成类的初始化。例如,当执行 Person person = new Person();
时,JVM 会先确认 Person
类是否已加载。
类加载完成后,JVM 会为新对象分配内存空间。分配内存的方式主要有两种:
选择哪种分配方式由 Java 堆是否规整决定,而 Java 堆是否规整又由所采用的垃圾收集器是否带有空间压缩整理(Compact)的能力决定。
内存分配完成后,JVM 会将分配到的内存空间都初始化为零值(不包括对象头)。这一步操作保证了对象的实例字段在 Java 代码中可以不赋初始值就直接使用,程序能访问到这些字段的数据类型所对应的零值。例如,int
类型的字段初始值为 0,boolean
类型的字段初始值为 false
。
JVM 会对对象进行必要的设置,例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希码、对象的 GC 分代年龄等信息。这些信息存放在对象的对象头(Object Header)之中。
init
方法在上述工作都完成之后,从 Java 程序的视角看来,对象已经产生了,但从 JVM 的视角来看,对象创建才刚刚开始——<init>
方法还没有执行,所有的字段都还为零。所以一般来说(由字节码中是否跟随有 invokespecial
指令所决定),执行 new
指令之后会接着执行 <init>
方法,把对象按照程序员的意愿进行初始化,这样一个真正可用的对象才算完全被构造出来。
在 Java 中,对象的销毁主要由垃圾回收机制(GC)负责。JVM 会通过可达性分析算法来判断对象是否存活。该算法以一系列被称为“GC Roots”的对象为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链,当一个对象到 GC Roots 没有任何引用链相连时,则证明此对象是不可达的,就有可能被回收。常见的 GC Roots 包括:
当对象被判定为不可达时,它会被第一次标记。但此时对象还不会被立即回收,而是会被放入一个名为 F - Queue
的队列中,并由一个低优先级的线程去执行队列中对象的 finalize()
方法(如果对象重写了该方法)。
finalize()
方法执行finalize()
方法是对象逃脱死亡命运的最后一次机会。在该方法中,对象可以重新与引用链上的任何一个对象建立关联,例如把自己(this
关键字)赋值给某个类变量或者对象的成员变量。如果对象在 finalize()
方法中成功拯救了自己,那在第二次标记时它将被移除出“即将回收”的集合;如果对象没有逃脱,那基本上它就真的要被回收了。不过需要注意的是,finalize()
方法的执行是不可靠的,JVM 并不保证该方法一定会被执行。
如果对象在 finalize()
方法执行后仍然没有与 GC Roots 建立引用关系,它会被进行第二次标记。经过第二次标记的对象,就会被真正地列入可回收对象的集合。
当垃圾回收器执行垃圾回收操作时,会回收那些经过第二次标记的对象所占用的内存空间,将其释放回 Java 堆中,供后续新对象的分配使用。不同的垃圾回收器采用不同的算法和策略来执行垃圾回收,例如标记 - 清除算法、标记 - 整理算法、复制算法等。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。