Garbage Collection

以下為Java記憶體管理的幾個要項…
1.      在Java的程式語言規格之中並未詳細說明殘餘記憶體收集機制的法則,因此每一家JVM的開發廠商都根據自己的演算法來研發殘餘記憶體收集機制。
2.      殘餘記憶體是以單獨的執行緒來執行的。
3.      通常JVM會在記憶體不足時啟動殘餘記憶體收集的執行緒。
4.      程式設計師可以透過以下兩種方法來”通知”JVM啟動殘餘記憶體收集:
Runtime.getRuntime().gc();

System.gc();

當程式設計師以上述兩種方法通知JVM啟動殘餘記憶體收集時,並不能保證JVM會啟動殘餘記憶體收集,事實上JVM只是在以上方法呼叫時”儘可能”執行殘餘記憶體收集。
5.      要確保不需要的物件會被殘餘記憶體機制收集掉的話,必須移除該物件的所有參考,移除參考的方法很簡單,如下… (將物件設為null即可)
MyObj m = new MyObj();
m.doSomethingUseful();
m = null;


6.      當一個method執行完離開它的區間時,所有區域物件都會自動被設為null,等待被回收。如下程式,當程式離開doSomething時,會產生三個垃圾物件,等待被回收。 (三個為陣列本身及陣列的兩個元素)

public class Question08 {
        public static void main(String[] args) {
               Question08 q08 = new Question08();
               q08.doSomething();     //line 1
               Thread.sleep(20000);
        }
 
        public void doSomething(){
               Object[] objArray = new Object[2];
               for(int i=0;i<objArray.length;i++){
                       objArray[i] = new Object();
               }
        }
}


7.      殘餘記憶體收集機制有一項問題,就是它只復原記憶體,但是Java物件除了佔用記憶體之外,也可能佔用系統資源,如下程式,說明Java程式可能在產生exception時,沒有將檔案關閉…
public void readData(String filename) {
   FileInputStream fis;
   try {
     fis = new FileInputStream(filename);
    fis.close();
     fis = null;
  }
  catch(IOException e) { }
}

為了處理這種狀況,Java提供了finalize的機制,在Object類別中有如下method…
protected void finalize() throws Throwable { }

JVM會在某物件沒有其它參考時執行該物件的finalize,而在實際收回物件所佔用的記憶體之前將系統資源復原,但需注意的是,Java語言規格中並未規定finalize一定要在殘餘記憶體收集的執行緒將物件丟棄之前立即執行,因此對程式設計師而言,還是無法保證finalize一定會執行,基於這樣的不確定因素,finalize被視為清除物件所佔用系統資源的”最後機會”,而不是復原系統資源的正常做法。

8.      finalize是可以被override的,在上述程式你可以這麼override finalize…
protected void finalize() throws Throwable {
  if (fis != null) {
    fis.close();
    super.finalize();
  }
}

需注意,收回系統資源之後,最好再呼叫上一層的finalize,以確保其它未被回收的系統資源被回收。

9.      雖然finalize有可能拋出exception,但是JVM會忽略該方法所拋出的exception。

ref:http://my.so-net.net.tw/idealist/Java/Basic/Garbage.html

你可能感兴趣的:(java,jvm,thread,.net,算法)