納尼,Java 存在內存泄泄泄泄泄泄漏嗎?

科技伍小黑 發佈 2020-01-03T03:54:23+00:00

Java最牛逼的一個特性就是垃圾回收機制,不用像 C++ 需要手動管理內存,所以作為 Java 程式設計師很幸福,只管 New New New 即可,反正 Java 會自動回收過期的對象。

01. 怎麼回事?

納尼,Java 不是自動管理內存嗎?怎麼可能會出現內存泄泄泄泄泄泄漏!

Java 最牛逼的一個特性就是垃圾回收機制,不用像 C++ 需要手動管理內存,所以作為 Java 程式設計師很幸福,只管 New New New 即可,反正 Java 會自動回收過期的對象。。。

那麼 Java 都自動管理內存了,那怎麼會出現內存泄漏,難道 Jvm 有 bug? 不要急,且聽我慢慢道來。。

02. 怎麼判斷可以被回收

先了解一下 Jvm 是怎麼判斷一個對象可以被回收。一般有兩種方式,一種是引用計數法,一種是可達性分析。

引用計數法:每個對象有一個引用計數屬性,新增一個引用時計數加1,引用釋放時計數減1,計數為0時可以回收。

這個辦法看起來挺簡單的,但是如果出現 A 引用了 B,B 又引用了 A,這時候就算他們都不再使用了,但因為相互引用 計算器=1 永遠無法被回收。

此方法簡單,無法解決對象相互循環引用的問題。

可達性分析(Reachability Analysis):從 GC Roots 開始向下搜索,搜索所走過的路徑稱為引用鏈。當一個對象到 GC Roots 沒有任何引用鏈相連時,則證明此對象是不可用的,那麼虛擬機就判斷是可回收對象。

可達性分析可以解決循環引用的問題。

那麼 gc roots 對象是哪些呢

  • 虛擬機棧中引用的對象
  • 方法區中類靜態屬性引用的對象
  • 方法區中常量引用的對象
  • 本地方法棧中JNI[即一般說的Native]引用的對象

目前主流的虛擬機中大多使用可達性分析的方式來判定對象是否可被 GC 回收。

03. 什麼情況下會出現內存泄漏

既然可達性分析好像已經很牛逼的樣子了,怎麼可能還會出現內存泄漏呢,那我們再來看一下內存泄漏的定義。

內存泄露就是指一個不再被程序使用的對象或變量一直被占據在內存中。

有可能此對象已經不使用了,但是還有其它對象保持著此對象的引用,就會導致 GC 不能回收此對象,這種情況下就會出現內存泄漏。

寫一個程序讓出現內存泄漏

①長生命周期的對象持有短生命周期對象的引用就很可能發生內存泄露,儘管短生命周期對象已經不再需要,但是因為長生命周期對象持有它的引用而導致不能被回收。

public class Simple {
    Object object;
    public void method1(){
        object = new Object();
        //...其他代碼
    }
}

這裡的 object 實例,其實我們期望它只作用於 method1() 方法中,且其他地方不會再用到它,但是,當method1()方法執行完成後,object 對象所分配的內存不會馬上被認為是可以被釋放的對象,只有在 Simple 類創建的對象被釋放後才會被釋放,嚴格的說,這就是一種內存泄露。

解決方法就是將 object 作為 method1() 方法中的局部變量。

public class Simple {
    Object object;
    public void method1(){
        object = new Object();
        //...其他代碼
        object = null;
    }
}

當然大家有可能會想就這一個方法也不會有多大影響,但如果在某些項目中,一個方法在一分鐘之內調用上萬次的時候,就會出現很明顯的內存泄漏現象。

②集合中的內存泄漏,比如 HashMap、ArrayList 等,這些對象經常會發生內存泄露。比如當它們被聲明為靜態對象時,它們的生命周期會跟應用程式的生命周期一樣長,很容易造成內存不足。

下面給出了一個關於集合內存泄露的例子。

Vector v=new Vector(10);
for (int i=1;i<100; i++)
{
    Object o=new Object();
    v.add(o);
    o=null;
}
//此時,所有的Object對象都沒有被釋放,因為變量v引用這些對象。

在這個例子中,我們循環申請 Object 對象,並將所申請的對象放入一個 Vector 中,如果我們僅僅釋放引用本身,那麼 Vector 仍然引用該對象,所以這個對象對 GC 來說是不可回收的。

因此,如果對象加入到 Vector 後,還必須從 Vector 中刪除,最簡單的方法就是將 Vector 對象設置為 null。

以上兩種是最常見的內存泄漏案例。當然還有一些內存泄漏的例子,這裡就不再一一例舉了,感興趣的同學可以在網上找找資料。

04. 內存泄漏和內存溢出

很多同學總是搞不清楚,內存泄漏和內存溢出的區別,它倆是兩個完全不同的概念, 它們之間存在一些關聯。

內存溢出 out of memory,是指程序在申請內存時,沒有足夠的內存空間供其使用,出現 out of memory;

內存泄露 memory leak,是指程序在申請內存後,無法釋放已申請的內存空間,一次內存泄露危害可以忽略,但內存泄露堆積後果很嚴重,無論多少內存,遲早會被占光。

所以內存泄漏可能會導致內存溢出,但內存溢出並不完全都是因為內存泄漏,也有可能使用了太多的大對象導致。

05. 如何檢測內存泄漏

最後一個重要的問題,就是如何檢測 Java 的內存泄漏。目前,我們通常使用一些工具來檢查 Java 程序的內存泄漏問題。

市場上已有幾種專業檢查 Java 內存泄漏的工具,它們的基本工作原理大同小異,都是通過監測 Java 程序運行時,所有對象的申請、釋放等動作,將內存管理的所有信息進行統計、分析、可視化。開發人員將根據這些信息判斷程序是否有內存泄漏問題。

這些工具包括 Plumbr 、Eclipse Memory Analyzer、JProbe Profiler、JVisualVM 等。

關鍵字: