程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> JAVA編程入門知識 >> 深入java垃圾回收的詳解

深入java垃圾回收的詳解

編輯:JAVA編程入門知識

1.垃圾收集算法的核心思想
Java語言建立了垃圾收集機制,用以跟蹤正在使用的對象和發現並回收不再使用(引用)的對象。該機制可以有效防范動態內存分配中可能發生的兩個危險:因內存垃圾過多而引發的內存耗盡,以及不恰當的內存釋放所造成的內存非法引用。

垃圾收集算法的核心思想是:對虛擬機可用內存空間,即堆空間中的對象進行識別,如果對象正在被引用,那麼稱其為存活對象,反之,如果對象不再被引用,則為垃圾對象,可以回收其占據的空間,用於再分配。垃圾收集算法的選擇和垃圾收集系統參數的合理調節直接影響著系統性能,因此需要開發人員做比較深入的了解。

2.觸發主GC(Garbage Collector)的條件
JVM進行次GC的頻率很高,但因為這種GC占用時間極短,所以對系統產生的影響不大。更值得關注的是主GC的觸發條件,因為它對系統影響很明顯。總的來說,有兩個條件會觸發主GC:

(1)當應用程序空閒時,即沒有應用線程在運行時,GC會被調用。因為GC在優先級最低的線程中進行,所以當應用忙時,GC線程就不會被調用,但以下條件除外。

(2)Java堆內存不足時,GC會被調用。當應用線程在運行,並在運行過程中創建新對象,若這時內存空間不足,JVM就會強制地調用GC線程,以便回收內存用於新的分配。若GC一次之後仍不能滿足內存分配的要求,JVM會再進行兩次GC作進一步的嘗試,若仍無法滿足要求,則 JVM將報“out of memory”的錯誤,Java應用將停止。

由於是否進行主GC由JVM根據系統環境決定,而系統環境在不斷的變化當中,所以主GC的運行具有不確定性,無法預計它何時必然出現,但可以確定的是對一個長期運行的應用來說,其主GC是反復進行的。

3.減少GC開銷的措施
根據上述GC的機制,程序的運行會直接影響系統環境的變化,從而影響GC的觸發。若不針對GC的特點進行設計和編碼,就會出現內存駐留等一系列負面影響。為了避免這些影響,基本的原則就是盡可能地減少垃圾和減少GC過程中的開銷。具體措施包括以下幾個方面:

(1)不要顯式調用System.gc()
此函數建議JVM進行主GC,雖然只是建議而非一定,但很多情況下它會觸發主GC,從而增加主GC的頻率,也即增加了間歇性停頓的次數。這裡特別需要說明的是,在代碼中顯示的調用System.gc(),並不一定能夠進行GC,這個我們可以通過finalize()方法進行驗證,即主動調用System.gc(),並不一定每次都調用finalize()方法。finalize()方法的特征是在對象被回收之前, 首先調用finalize()方法。

(2)盡量減少臨時對象的使用
臨時對象在跳出函數調用後,會成為垃圾,少用臨時變量就相當於減少了垃圾的產生,從而延長了出現上述第二個觸發條件出現的時間,減少了主GC的機會。

(3)對象不用時最好顯式置為Null
一般而言,為Null的對象都會被作為垃圾處理,所以將不用的對象顯式地設為Null,有利於GC收集器判定垃圾,從而提高了GC的效率。

(4)盡量使用StringBuffer,而不用String來累加字符串(詳見blog另一篇文章JAVA中String與StringBuffer)
由於String是固定長的字符串對象,累加String對象時,並非在一個String對象中擴增,而是重新創建新的String對象,如 Str5=Str1+Str2+Str3+Str4,這條語句執行過程中會產生多個垃圾對象,因為對次作“+”操作時都必須創建新的String對象,但這些過渡對象對系統來說是沒有實際意義的,只會增加更多的垃圾。避免這種情況可以改用StringBuffer來累加字符串,因StringBuffer 是可變長的,它在原有基礎上進行擴增,不會產生中間對象。

(5)能用基本類型如Int,Long,就不用Integer,Long對象
基本類型變量占用的內存資源比相應對象占用的少得多,如果沒有必要,最好使用基本變量。什麼情況下需要使用Integer?

(6)盡量少用靜態對象變量
靜態變量屬於全局變量,不會被GC回收,它們會一直占用內存。

(7)分散對象創建或刪除的時間
集中在短時間內大量創建新對象,特別是大對象,會導致突然需要大量內存,JVM在面臨這種情況時,只能進行主GC,以回收內存或整合內存碎片, 從而增加主GC的頻率。集中刪除對象,道理也是一樣的。它使得突然出現了大量的垃圾對象,空閒空間必然減少,從而大大增加了下一次創建新對象時強制主GC 的機會。

4.垃圾回收算法
(1)引用計數收集器
引用計數是垃圾收集的早期策略。在這種方法中,堆中每一個對象都有一個引用計數。當一個對象被創建了,並且指向該對象的引用被分配給一個變量,這個對象的引用計數被設置為1。比如新建一個對象A a=new A();然後a被分配給另外一個變量b,也就是b=a;那麼對象a的引用計數+1。當任何其他變量被賦值為對這個對象的引用時,計數加1。當一個對象的引用超過生存期或者被設置一個新的值時,對象的引用計數減1,比如令b=c,則a的引用計數-1。任何引用計數為0的對象可以被當做垃圾收集。當一個對象被垃圾收集的時候,它引用的任何對象計數減1。在這種方法中,一個對象被垃圾收集後可能導致後續其他對象的垃圾收集行動。比如A a=new A();b=a;當b被垃圾回收以後,a的引用計數變為0,這樣導致a也被垃圾回收。

方法的好處:引用計數收集器可以很快執行,交織在程序的運行之中。這個提醒對於程序不能被長時間打斷的實時環境很有利。

方法的壞處:引用計數無法檢測出循環(即兩個或者更多的對象互相引用)。循環的例子如,父對象有一個子對象的引用,子對象又反過來引用父對象。這樣對象用戶都不可能計數為0,就算它們已經無法被執行程序的根對象觸及。還有一個壞處就是,每次引用計數的增加或者減少都帶來額外的開銷。

(2)追蹤收集器
垃圾檢測通常通過建立一個根對象的集合並且檢查從這些根對象開始的可觸及性來實現。如果正在執行的程序可以訪問到的根對象和某個對象之間存在引用路徑,這個對象就是可觸及的。對於程序來說,根對象總是可以訪問的。從這些根對象開始,任何可以被觸及的對象都是被認為是“活動”的對象。無法被觸及的對象被認為是垃圾,因為它們不在影響程序的未來執行。

跟蹤收集器是追蹤從根結點開始的對象引用圖。在追蹤過程中遇到的對象以某手方式打上標記。總的來說,要麼在對象本身上設置標記,要麼用一個獨立的位圖來設置標記。當追蹤結束時,未被標記的對象就是無法觸及的,從而可以被收集。

基本的追蹤算法被稱作“標記並清除”。這個名字指出垃圾手機的兩個階段。在標記階段,垃圾收集器遍歷引用樹,標記每一個遇到的對象。在清除階段,未被標記的對象被釋放,釋放對象後獲得的內存被返回到正在執行的程序。在Java虛擬機中,清除步驟必須包括對象的終結。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved