Java內存泄漏相關

 

以前學習了javaGC的原理機制,有了必定的瞭解,如今作一個整理總結,便於理解記憶,包括三個問題:html

1. java GC是何時作的?java

2. java GC做用的東西是什麼?程序員

3. java GC具體都作了些什麼事情?算法

關於java GC原理參看另外一篇隨筆:數據庫

http://www.cnblogs.com/clarke157/p/7091990.html編程

 

1. java GC是何時作的?網絡

也就是GC的觸發條件,eden 滿了minor gc,升到老年代的對象大於老年代剩餘空間full gc,或者小於時被HandlePromotionFailure參數強制full gc;gc與非gc時間耗時超過了GCTimeRatio的限制引起OOM,調優諸如經過NewRatio控制新生代老年代比例,經過 MaxTenuringThreshold控制進入老年前生存次數等。jvm

2. java GC做用的東西是什麼?socket

從GC root搜索不到,並且通過第一次標記、清理後,仍然沒有復活的對象。學習

3. java GC具體都作了些什麼事情?

年輕代作的是複製清理、from survivor、to survivor是幹啥用的、年老代作的是標記清理、標記清理後碎片要不要整理、複製清理和標記清理有有什麼優劣勢

年輕代上的內存分配是這樣的,年輕代能夠分爲3個區域:Eden區和兩個存活區(Survivor 0 、Survivor 1)。

 

  1. 絕大多數剛建立的對象會被分配在Eden區,其中的大多數對象很快就會消亡。Eden區是連續的內存空間,所以在其上分配內存極快;

  2. 當Eden區滿的時候,執行Minor GC,將消亡的對象清理掉,並將剩餘的對象複製到一個存活區Survivor0(此時,Survivor1是空白的,兩個Survivor總有一個是空白的);

  3. 此後,每次Eden區滿了,就執行一次Minor GC,並將剩餘的對象都添加到Survivor0;

  4. 當Survivor0也滿的時候,將其中仍然活着的對象直接複製到Survivor1,之後Eden區執行Minor GC後,就將剩餘的對象添加Survivor1(此時,Survivor0是空白的)。

  5. 當兩個存活區切換了幾回(HotSpot虛擬機默認15次,用-XX:MaxTenuringThreshold控制,大於該值進入老年代)以後,仍然存活的對象(其實只有一小部分,好比,咱們本身定義的對象),將被複制到老年代。

Java 也是有內存泄露的,雖然Java有著名的GC( 垃圾回收機制),因爲Java是經過程序來分配內存空間,釋放則交給GC, 因爲程序編寫不當,仍然會引發內存泄露。

Java中的內存泄露的狀況:長生命週期的對象持有短生命週期對象的引用就極可能發生內存泄露,儘管短生命週期對象已經再也不須要,可是由於長生命周 期對象持有它的引用而致使不能被回收,這就是java中內存泄露的發生場景。例如在Application中保存一個對象,這個對象其實沒有使用了,可是 因爲一直被引用,形成不能回收。
內存泄露的狀況是:一個外部類的實例對象的方法返回了一個內部類的實例對象,這個內部類對象被長期引用了,即便那個外部類實例對象再也不被使用,但因爲內部類持久外部類的實例對象,這個外部類對象將不會被垃圾回收。
內存泄露的另一種狀況:當一個對象被存儲進HashSet集合中之後,就不能修改這個對象中的那些參與計算哈希值的字段了,不然,對象修改後的哈希值與 最初存儲進HashSet集合中時的哈希值就不一樣了,這也會致使沒法從HashSet集合中單獨刪除當前對象,形成內存泄露。

 

Java內存泄露根本緣由是什麼呢?長生命週期的對象持有短生命週期對象的引用就極可能發生內存泄露,儘管短生命週期對象已經再也不須要,可是由於長生命週期對象持有它的引用而致使不能被回收,這就是java中內存泄露的發生場景。具體主要有以下幾大類: 
一、靜態集合類引發內存泄露: 
像HashMap、Vector等的使用最容易出現內存泄露,這些靜態變量的生命週期和應用程序一致,他們所引用的全部的對象Object也不能被釋放,由於他們也將一直被Vector等引用着。 
例: 
Static Vector v = new Vector(10); 
for (int i = 1; i<100; i++) 
{ 
Object o = new Object(); 
v.add(o); 
o = null; 
}// 
在這個例子中,循環申請Object 對象,並將所申請的對象放入一個Vector 中,若是僅僅釋放引用自己(o=null),那麼Vector 仍然引用該對象,因此這個對象對GC 來講是不可回收的。所以,若是對象加入到Vector 後,還必須從Vector 中刪除,最簡單的方法就是將Vector對象設置爲null。
二、當集合裏面的對象屬性被修改後,再調用remove()方法時不起做用。
例: 
public static void main(String[] args) 
{ 
Set<Person> set = new HashSet<Person>(); 
Person p1 = new Person("唐僧","pwd1",25); 
Person p2 = new Person("孫悟空","pwd2",26); 
Person p3 = new Person("豬八戒","pwd3",27); 
set.add(p1); 
set.add(p2); 
set.add(p3); 
System.out.println("總共有:"+set.size()+" 個元素!"); //結果:總共有:3 個元素! 
p3.setAge(2); //修改p3的年齡,此時p3元素對應的hashcode值發生改變 

set.remove(p3); //此時remove不掉,形成內存泄漏
set.add(p3); //從新添加,竟然添加成功 
System.out.println("總共有:"+set.size()+" 個元素!"); //結果:總共有:4 個元素! 
for (Person person : set) 
{ 
System.out.println(person); 
} 
}
三、監聽器 
在java 編程中,咱們都須要和監聽器打交道,一般一個應用當中會用到不少監聽器,咱們會調用一個控件的諸如addXXXListener()等方法來增長監聽器,但每每在釋放對象的時候卻沒有記住去刪除這些監聽器,從而增長了內存泄漏的機會。
四、各類鏈接 
好比數據庫鏈接(dataSourse.getConnection()),網絡鏈接(socket)和io鏈接,除非其顯式的調用了其close()方法將其鏈接關閉,不然是不會自動被GC 回收的。對於Resultset 和Statement 對象能夠不進行顯式回收,但Connection 必定要顯式回收,由於Connection 在任什麼時候候都沒法自動回收,而Connection一旦回收,Resultset 和Statement 對象就會當即爲NULL。可是若是使用鏈接池,狀況就不同了,除了要顯式地關閉鏈接,還必須顯式地關閉Resultset Statement 對象(關閉其中一個,另一個也會關閉),不然就會形成大量的Statement 對象沒法釋放,從而引發內存泄漏。這種狀況下通常都會在try裏面去的鏈接,在finally裏面釋放鏈接。
五、內部類和外部模塊等的引用 
內部類的引用是比較容易遺忘的一種,並且一旦沒釋放可能致使一系列的後繼類對象沒有釋放。此外程序員還要當心外部模塊不經意的引用,例如程序員A 負責A 模塊,調用了B 模塊的一個方法如: 
public void registerMsg(Object b); 
這種調用就要很是當心了,傳入了一個對象,極可能模塊B就保持了對該對象的引用,這時候就須要注意模塊B 是否提供相應的操做去除引用。
六、單例模式 
不正確使用單例模式是引發內存泄露的一個常見問題,單例對象在被初始化後將在JVM的整個生命週期中存在(以靜態變量的方式),若是單例對象持有外部對象的引用,那麼這個外部對象將不能被jvm正常回收,致使內存泄露,考慮下面的例子: 
class A{ 
public A(){ 
B.getInstance().setA(this); 
} 
.... 
} 
//B類採用單例模式 
class B{ 
private A a; 
private static B instance=new B(); 
public B(){} 
public static B getInstance(){ 
return instance; 
} 
public void setA(A a){ 
this.a=a; 
} 
//getter... 
} 
顯然B採用singleton模式,它持有一個A對象的引用,而這個A類的對象將不能被回收。想象下若是A是個比較複雜的對象或者集合類型會發生什麼狀況
對象A和B互相引用,最後會不會被GC回收?GC裏邊在JVM當中是使用的ROOT算法,ROOT算法,什麼稱做爲ROOT呢,就是說類的靜態成員,靜態成員就是static修飾的那種,是「根」的 一個,根還包括方法中的成員變量,只有成員或對象不掛在根上,GC的時候就可能把他們搞掉,這裏提到的循環引用,就看這個循環引用是否掛在根上,若是掛在 根上,若是這個根還被JVM的Java代碼所執行的話,就不會GC掉,若是說這個根已經被釋放掉了,這個對象不掛在跟上了,那個這個對象就會被GC掉。參考連接:https://www.cnblogs.com/clarke157/p/7903660.html
相關文章
相關標籤/搜索