如果对象的引用被置为null
,垃圾收集器是否会立即释放对象占用的内存?
不会,在下一个垃圾回收周期中,这个对象将是可被回收的。
也就是说当一个对象的引用变为 null
时,并不会被垃圾收集器立刻回收,而是在下一次垃圾回收时才会释放其占用的内存。
finalize()
方法工作原理
一旦垃圾回收器准备好释放对象占用的存储空间,将首先调用其 finalize()
方法(如果如果覆盖了finalize()
),并且在下一次垃圾回收动作发生时,才会真正回收对象占用的内存。
至于为什么在下一次垃圾回收动作发生时才会回收内存,原因是如果一个对象覆盖了 finalize()
方法,那么在真正被宣告死亡的时候,至少需要经过两次标记。第一次被标记的时候会被放在 一个 F-Queue
队列中,finalize()
方法是对象逃脱死亡命运的最后一次机会。在第二次标记的时候,如果该对象成功与引用链(GC-Roots
)上的任何一个对象关联,那么它仍然可以存活下来,否则将会被垃圾收集器回收。
Java 8 的内存分代改进
在 jdk1.8 中对内存模型中方法区的实现永久代(Perm Gen space)进行了移除,取而代之的是元空间(Metaspace)。
原因是在方法区中实现垃圾回收的条件比较苛刻,因此存在着内存溢出的风险。在 jdk1.8 之后,当方法区内存使用较多时,元空间会使用物理内存,减少了风险。如下
分别写出堆内存溢出与栈内存溢出的程序
这里涉及到Java 运行时数据区域 方面的知识。
栈内存溢出
public void f() {
f();
}
堆内存溢出