Java 与 C 的其中一个差别就在於写 C 程式时,假如我们 new 了一个物件,同时我们也必须下 delete 来清除它, C 语言的核心并不会判定这个物件是否不再使用,而在程式执行时,需不需要将物件移除,也全都由使用者自行控制。其实这个原意很好,但是写到大型专案,或是 Multi-Thread 程式时,往往这些物件会变成难以控制,有时候我们甚至会试图去使用一个已经不存在的指标。
在 Java 中, JVM 帮我们治理这一层,因此当物件不在被使用时, JV M 会在"适当"的时间点将它由记忆体中移除。
所有的物件均继续 Object ,因此 Object 这个物件中,有一个很有用的 method 叫做 finalize() ,这一个 method 依照定义,当 Garbage Collection 要将这个物件移除前会先呼叫这个物件的 finalize() 。也就是说,当我们在实作我们的 Class 的时候,最好将这一个 method override 。这样一来,当 Garbage Collection 要清除物件的时候,就会执行到我们自己写的 finalize()。
因此我们设计了底下这个物件,这个物件在我们建立时,需要带入一个 int 值,这一个 int 值,而当 finalize() 被执行的时候,他会将这个 int 值列印出来,好让我们知道 Garbage Collection正预备清除哪一个物件。
public class testobject
{
private int itell;
public testobject()
{
this.itell = 0;
}
public testobject(int i)
{
this.itell = i;
}
public void finalize()
{
System.out.println("testobject id = "+itell);
}
}
接著,我写了底下这一个 testfinalize.java ,这一个 testfinalize.java 会宣告一个阵列,这一个阵列中,会存放 testobject 物件,届时 我会将这一个阵列设为 null ,表示我不再使用这一个阵列了。
public class testfinalize
{
public static void main(String argv[])
{
testobject [] testo = new testobject[2];
testo[0] = new testobject(1);
testo[1] = new testobject(2);
testo = null;
System.out.println("finish...");
}
}
执行的结果是
C: empjavajava testfinalize
finish...
C: empjava
希奇,是 garbage collection 没有执行 finalize() 吗?? 不是的,是 garbage collection 根本没有被启动,在前面说过,garbage collection 会在适当的时机启动,而这一个 testfinalize 程式很小, JVM 认为不需要执行 Garbage Collection ,因此程式结束,记忆体释放。
是不是这样就没办法知道 garbage collection 做了时么事情??不是我们还可以透过两个 method 来建议 JVM 执行 garbage collection,一个是 System.gc(); ,另外一个建议的方式是 Runtime.getRuntime().gc(), System.gc() 也会执行 Runtime.getRuntime().gc() ,不论哪一个,他们都会建议 JVM 去执行另外一个 Thread ,这一个 Thread 会扫瞄所有没有被用到的物件,将他们清除,并且取得他们所占用的记忆体。