Java內(nèi)存泄露

1. Java內(nèi)存回收機制

原文轉(zhuǎn)自:Java 內(nèi)存泄露

不論哪種語言的內(nèi)存分配方式门坷,都需要返回所分配內(nèi)存的真實地址宣鄙,也就是返回一個指針到內(nèi)存塊的首地址。Java中對象是采用new或者反射的方法創(chuàng)建的默蚌,這些對象的創(chuàng)建都是在堆(Heap)中分配的冻晤,所有對象的回收都是由Java虛擬機通過垃圾回收機制完成的。GC為了能夠正確釋放對象绸吸,會監(jiān)控每個對象的運行狀況鼻弧,對他們的申請、引用锦茁、被引用攘轩、賦值等狀況進行監(jiān)控,Java會使用有向圖的方法進行管理內(nèi)存码俩,實時監(jiān)控對象是否可以達到度帮,如果不可到達,則就將其回收稿存,這樣也可以消除引用循環(huán)的問題笨篷。在Java語言中,判斷一個內(nèi)存空間是否符合垃圾收集標準有兩個:一個是給對象賦予了空值null瓣履,以下再沒有調(diào)用過率翅,另一個是給對象賦予了新值,這樣重新分配了內(nèi)存空間袖迎。

2. Java內(nèi)存泄漏引起的原因

內(nèi)存泄漏是指無用對象(不再使用的對象)持續(xù)占有內(nèi)存或無用對象的內(nèi)存得不到及時釋放冕臭,從而造成內(nèi)存空間的浪費稱為內(nèi)存泄漏。內(nèi)存泄露有時不嚴重且不易察覺瓢棒,這樣開發(fā)者就不知道存在內(nèi)存泄露浴韭,但有時也會很嚴重丘喻,會提示你Out of memory脯宿。

Java內(nèi)存泄漏的根本原因是什么呢?長生命周期的對象持有短生命周期對象的引用就很可能發(fā)生內(nèi)存泄漏泉粉,盡管短生命周期對象已經(jīng)不再需要连霉,但是因為長生命周期持有它的引用而導致不能被回收,這就是Java中內(nèi)存泄漏的發(fā)生場景嗡靡。具體主要有如下幾大類:

①靜態(tài)集合類引起內(nèi)存泄漏:

像HashMap跺撼、Vector等的使用最容易出現(xiàn)內(nèi)存泄露,這些靜態(tài)變量的生命周期和應用程序一致讨彼,他們所引用的所有的對象Object也不能被釋放歉井,因為他們也將一直被Vector等引用著。

例如:

Static Vector v = new Vector(10);
for (int i = 1; i<100; i++){
    Object o = new Object();
    v.add(o);
    o = null;
}

在這個例子中哈误,循環(huán)申請Object 對象哩至,并將所申請的對象放入一個Vector 中躏嚎,如果僅僅釋放引用本身(o=null),那么Vector 仍然引用該對象菩貌,所以這個對象對GC 來說是不可回收的卢佣。因此,如果對象加入到Vector 后箭阶,還必須從Vector 中刪除虚茶,最簡單的方法就是將Vector對象設置為null。

②當集合里面的對象屬性被修改后仇参,再調(diào)用remove()方法時不起作用:

注意:定義的對象中要實現(xiàn)equals() , hashCode() 方法嘹叫。

例如:

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()+" 個元素!"); //結(jié)果:總共有:3 個元素!
    p3.setAge(2); //修改p3的年齡,此時p3元素對應的hashcode值發(fā)生改變

    set.remove(p3); //此時remove不掉,造成內(nèi)存泄漏

    set.add(p3); //重新添加诈乒,居然添加成功
    System.out.println("總共有:"+set.size()+" 個元素!"); //結(jié)果:總共有:4 個元素!
    for (Person person : set){
        System.out.println(person);
    }
}

③監(jiān)聽器:

在 Java 編程中待笑,我們都需要和監(jiān)聽器打交道,通常一個應用當中會用到很多監(jiān)聽器抓谴,我們會調(diào)用一個控件的諸如addXXXListener()等方法來增加監(jiān)聽器暮蹂,但往往在釋放對象的時候卻沒有記住去刪除這些監(jiān)聽器,從而增加了內(nèi)存泄漏的機會癌压。

④各種連接:

比如數(shù)據(jù)庫連接(dataSourse.getConnection())仰泻,網(wǎng)絡連接(socket)和io連接,除非其顯式的調(diào)用了其close()方法將其連接關閉滩届,否則是不會自動被GC 回收的集侯。對于Resultset 和Statement 對象可以不進行顯式回收,但Connection 一定要顯式回收帜消,因為Connection 在任何時候都無法自動回收棠枉,而Connection一旦回收,Resultset 和Statement 對象就會立即為NULL泡挺。但是如果使用連接池辈讶,情況就不一樣了,除了要顯式地關閉連接娄猫,還必須顯式地關閉Resultset Statement 對象(關閉其中一個贱除,另外一個也會關閉),否則就會造成大量的Statement 對象無法釋放媳溺,從而引起內(nèi)存泄漏月幌。這種情況下一般都會在try里面去的連接,在finally里面釋放連接悬蔽。

⑤內(nèi)部類和外部模塊的引用:

內(nèi)部類的引用是比較容易遺忘的一種扯躺,而且一旦沒釋放可能導致一系列的后繼類對象沒有釋放。此外程序員還要小心外部模塊不經(jīng)意的引用,例如程序員A 負責A 模塊录语,調(diào)用了B 模塊的一個方法如: public void registerMsg(Object b); 這種調(diào)用就要非常小心了轴术,傳入了一個對象,很可能模塊B就保持了對該對象的引用钦无,這時候就需要注意模塊B 是否提供相應的操作去除引用逗栽。

⑥單例模式

不正確使用單例模式是引起內(nèi)存泄漏的一個常見問題,單例對象在初始化后將在JVM的整個生命周期中存在(以靜態(tài)變量的方式)失暂,如果單例對象持有外部的引用彼宠,那么這個對象將不能被JVM正常回收弟塞,導致內(nèi)存泄漏凭峡,考慮下面的例子:

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類的對象將不能被回收摧冀。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市系宫,隨后出現(xiàn)的幾起案子索昂,更是在濱河造成了極大的恐慌,老刑警劉巖扩借,帶你破解...
    沈念sama閱讀 206,968評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件椒惨,死亡現(xiàn)場離奇詭異,居然都是意外死亡潮罪,警方通過查閱死者的電腦和手機康谆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來嫉到,“玉大人沃暗,你說我怎么就攤上這事『味瘢” “怎么了孽锥?”我有些...
    開封第一講書人閱讀 153,220評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長导而。 經(jīng)常有香客問我忱叭,道長隔崎,這世上最難降的妖魔是什么今艺? 我笑而不...
    開封第一講書人閱讀 55,416評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮爵卒,結(jié)果婚禮上虚缎,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好实牡,可當我...
    茶點故事閱讀 64,425評論 5 374
  • 文/花漫 我一把揭開白布陌僵。 她就那樣靜靜地躺著,像睡著了一般创坞。 火紅的嫁衣襯著肌膚如雪碗短。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,144評論 1 285
  • 那天题涨,我揣著相機與錄音偎谁,去河邊找鬼。 笑死纲堵,一個胖子當著我的面吹牛巡雨,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播席函,決...
    沈念sama閱讀 38,432評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼铐望,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了茂附?” 一聲冷哼從身側(cè)響起正蛙,我...
    開封第一講書人閱讀 37,088評論 0 261
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎营曼,沒想到半個月后跟畅,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,586評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡溶推,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,028評論 2 325
  • 正文 我和宋清朗相戀三年徊件,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蒜危。...
    茶點故事閱讀 38,137評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡虱痕,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出辐赞,到底是詐尸還是另有隱情部翘,我是刑警寧澤,帶...
    沈念sama閱讀 33,783評論 4 324
  • 正文 年R本政府宣布响委,位于F島的核電站新思,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏赘风。R本人自食惡果不足惜夹囚,卻給世界環(huán)境...
    茶點故事閱讀 39,343評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望邀窃。 院中可真熱鬧荸哟,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至劣砍,卻和暖如春惧蛹,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背刑枝。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評論 1 262
  • 我被黑心中介騙來泰國打工赊淑, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人仅讽。 一個月前我還...
    沈念sama閱讀 45,595評論 2 355
  • 正文 我出身青樓陶缺,卻偏偏與公主長得像,于是被迫代替她去往敵國和親洁灵。 傳聞我的和親對象是個殘疾皇子饱岸,可洞房花燭夜當晚...
    茶點故事閱讀 42,901評論 2 345

推薦閱讀更多精彩內(nèi)容