什么叫內(nèi)存溢出何暮?內(nèi)存泄漏的定義又是什么?使用中如何避免跨新?

內(nèi)存泄露是指無用對象(不再使用的對象)持續(xù)占有內(nèi)存或無用對象的內(nèi)存得不到及時釋放,從而造成的內(nèi)存空間的浪費(fèi)稱為內(nèi)存泄露贰军。

?一蟹肘、基本概念

內(nèi)存溢出:簡單地說內(nèi)存溢出就是指程序運(yùn)行過程中申請的內(nèi)存大于系統(tǒng)能夠提供的內(nèi)存俯树,導(dǎo)致無法申請到足夠的內(nèi)存,于是就發(fā)生了內(nèi)存溢出阳欲。

內(nèi)存泄漏:內(nèi)存泄漏指程序運(yùn)行過程中分配內(nèi)存給臨時變量,用完之后卻沒有被GC回收球化,始終占用著內(nèi)存,既不能被使用也不能分配給其他程序筒愚,于是就發(fā)生了內(nèi)存泄漏。

內(nèi)存溢出 out of memory巢掺,是指程序在申請內(nèi)存時劲蜻,沒有足夠的內(nèi)存空間供其使用,出現(xiàn)out of memory;

內(nèi)存泄露 memory leak先嬉,是指程序在申請內(nèi)存后,無法釋放已申請的內(nèi)存空間含懊,一次內(nèi)存泄露危害可以忽略,但內(nèi)存泄露堆積后果很嚴(yán)重鳄袍,無論多少內(nèi)存,遲早會被占光。

memory leak會最終會導(dǎo)致out of memory!

內(nèi)存泄露是指無用對象(不再使用的對象)持續(xù)占有內(nèi)存或無用對象的內(nèi)存得不到及時釋放樱哼,從而造成的內(nèi)存空間的浪費(fèi)稱為內(nèi)存泄露。內(nèi)存泄露有時不嚴(yán)重且不易察覺搅幅,這樣開發(fā)者就不知道存在內(nèi)存泄露,但有時也會很嚴(yán)重茄唐,會提示你Out of memory蝇更。

二呼盆、內(nèi)存溢出的常見情況

內(nèi)存溢出有以下幾種常見的情況:

1、java.lang.OutOfMemoryError: PermGen space (持久帶溢出)

我們知道jvm通過持久帶實(shí)現(xiàn)了java虛擬機(jī)規(guī)范中的方法區(qū)访圃,而運(yùn)行時常量池就是保存在方法區(qū)中的,因此發(fā)生這種溢出可能是運(yùn)行時常量池溢出况脆,或是由于程序中使用了大量的jar或class,使得方法區(qū)中保存的class對象沒有被及時回收或者class信息占用的內(nèi)存超過了配置的大小格了。

2徽鼎、java.lang.OutOfMemoryError: Java heap space (堆溢出)

發(fā)生這種溢出的原因一般是創(chuàng)建的對象太多,在進(jìn)行垃圾回收之前對象數(shù)量達(dá)到了最大堆的容量限制纬傲。

解決這個區(qū)域異常的方法一般是通過內(nèi)存映像分析工具對Dump出來的堆轉(zhuǎn)儲快照進(jìn)行分析,看到底是內(nèi)存溢出還是內(nèi)存泄漏叹括。如果是內(nèi)存泄漏,可進(jìn)一步通過工具查看泄漏對象到GC Roots的引用鏈净嘀,定位出泄漏代碼的位置,修改程序或算法;如果不存在泄漏挖藏,就是說內(nèi)存中的對象確實(shí)都還必須存活,那就應(yīng)該檢查虛擬機(jī)的堆參數(shù)-Xmx(最大堆大小)和-Xms(初始堆大小)厢漩,與機(jī)器物理內(nèi)存對比看是否可以調(diào)大。

3溜嗜、虛擬機(jī)棧和本地方法棧溢出

如果線程請求的棧深度大于虛擬機(jī)所允許的最大深度,將拋出StackOverflowError辟躏。

如果虛擬機(jī)在擴(kuò)展棧時無法申請到足夠的內(nèi)存空間土全,則拋出OutOfMemoryError捎琐。

三、內(nèi)存泄漏

內(nèi)存泄漏的根本原因是長生命周期的對象持有短生命周期對象的引用瑞凑,盡管短生命周期的對象已經(jīng)不再需要,但由于長生命周期對象持有它的引用而導(dǎo)致不能被回收拨黔。

以發(fā)生的方式來分類,內(nèi)存泄漏可以分為4類:

1篱蝇、常發(fā)性內(nèi)存泄漏。發(fā)生內(nèi)存泄漏的代碼會被多次執(zhí)行到零截,每次被執(zhí)行的時候都會導(dǎo)致一塊內(nèi)存泄漏。

2涧衙、偶發(fā)性內(nèi)存泄漏。發(fā)生內(nèi)存泄漏的代碼只有在某些特定環(huán)境或操作過程下才會發(fā)生雁比。常發(fā)性和偶發(fā)性是相對的。對于特定的環(huán)境偎捎,偶發(fā)性的也許就變成了常發(fā)性的序攘。所以測試環(huán)境和測試方法對檢測內(nèi)存泄漏至關(guān)重要茴她。

3丈牢、一次性內(nèi)存泄漏。發(fā)生內(nèi)存泄漏的代碼只會被執(zhí)行一次己沛,或者由于算法上的缺陷距境,導(dǎo)致總會有一塊僅且一塊內(nèi)存發(fā)生泄漏。比如肮疗,在類的構(gòu)造函數(shù)中分配內(nèi)存扒接,在析構(gòu)函數(shù)中卻沒有釋放該內(nèi)存们衙,所以內(nèi)存泄漏只會發(fā)生一次碱呼。

4、隱式內(nèi)存泄漏愚臀。程序在運(yùn)行過程中不停的分配內(nèi)存,但是直到結(jié)束的時候才釋放內(nèi)存姑裂。嚴(yán)格的說這里并沒有發(fā)生內(nèi)存泄漏,因為最終程序釋放了所有申請的內(nèi)存舶斧。但是對于一個服務(wù)器程序,需要運(yùn)行幾天泽台,幾周甚至幾個月,不及時釋放內(nèi)存也可能導(dǎo)致最終耗盡系統(tǒng)的所有內(nèi)存怀酷。所以嗜闻,我們稱這類內(nèi)存泄漏為隱式內(nèi)存泄漏。

從用戶使用程序的角度來看泞辐,內(nèi)存泄漏本身不會產(chǎn)生什么危害,作為一般的用戶咐吼,根本感覺不到內(nèi)存泄漏的存在。真正有危害的是內(nèi)存泄漏的堆積厢塘,這會最終消耗盡系統(tǒng)所有的內(nèi)存。從這個角度來說晚碾,一次性內(nèi)存泄漏并沒有什么危害,因為它不會堆積格嘁,而隱式內(nèi)存泄漏危害性則非常大廊移,因為較之于常發(fā)性和偶發(fā)性內(nèi)存泄漏它更難被檢測到探入。

下面總結(jié)幾種常見的內(nèi)存泄漏:

1懂诗、靜態(tài)集合類引起的內(nèi)存泄漏:

像HashMap蜂嗽、Vector等的使用最容易出現(xiàn)內(nèi)存泄露,這些靜態(tài)變量的生命周期和應(yīng)用程序一致植旧,他們所引用的所有的對象Object也不能被釋放离唐,從而造成內(nèi)存泄漏,因為他們也將一直被Vector等引用著胖喳。

Vector<Object>?v=new?Vector<Object>(100);?

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對象設(shè)置為null。

2段只、修改HashSet中對象的參數(shù)值鉴扫,且參數(shù)是計算哈希值的字段

當(dāng)一個對象被存儲到HashSet集合中以后赞枕,修改了這個對象中那些參與計算哈希值的字段后,這個對象的哈希值與最初存儲在集合中的就不同了,這種情況下鹦赎,用contains方法在集合中檢索對象是找不到的谍椅,這將會導(dǎo)致無法從HashSet中刪除當(dāng)前對象,造成內(nèi)存泄漏古话,舉例如下:

public?static?void?main(String[]?args){?

Set<Person>?set?=?new?HashSet<Person>();?

Person?p1?=?new?Person("張三","1",25);?

Person?p2?=?new?Person("李四","2",26);?

Person?p3?=?new?Person("王五","3",27);?

set.add(p1);?

set.add(p2);?

set.add(p3);?

System.out.println("總共有:"+set.size()+"?個元素!");?//結(jié)果:總共有:3?個元素!?

?p3.setAge(2);?//修改p3的年齡,此時p3元素對應(yīng)的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);?

?}?

}?

3陪踩、監(jiān)聽器

在java 編程中,我們都需要和監(jiān)聽器打交道悉抵,通常一個應(yīng)用當(dāng)中會用到很多監(jiān)聽器肩狂,我們會調(diào)用一個控件的諸如addXXXListener()等方法來增加監(jiān)聽器,但往往在釋放對象的時候卻沒有記住去刪除這些監(jiān)聽器姥饰,從而增加了內(nèi)存泄漏的機(jī)會傻谁。

4、各種連接

比如數(shù)據(jù)庫連接(dataSourse.getConnection())列粪,網(wǎng)絡(luò)連接(socket)和io連接审磁,除非其顯式的調(diào)用了其close() 方法將其連接關(guān)閉,否則是不會自動被GC 回收的岂座。對于Resultset 和Statement 對象可以不進(jìn)行顯式回收态蒂,但Connection 一定要顯式回收饼齿,因為Connection 在任何時候都無法自動回收径密,而Connection一旦回收,Resultset 和Statement 對象就會立即為NULL鸳址。但是如果使用連接池稿黍,情況就不一樣了闻察,除了要顯式地關(guān)閉連接,還必須顯式地關(guān)閉Resultset Statement 對象(關(guān)閉其中一個,另外一個也會關(guān)閉)鸯乃,否則就會造成大量的Statement 對象無法釋放,從而引起內(nèi)存泄漏鸟悴。這種情況下一般都會在try里面去連接,在finally里面釋放連接震贵。

5水评、單例模式

如果單例對象持有外部對象的引用寇甸,那么這個外部對象將不能被jvm正衬妹梗回收友浸,導(dǎo)致內(nèi)存泄露。

不正確使用單例模式是引起內(nèi)存泄露的一個常見問題祭往,單例對象在被初始化后將在JVM的整個生命周期中存在(以靜態(tài)變量的方式)驮肉,如果單例對象持有外部對象的引用离钝,那么這個外部對象將不能被jvm正陈芽剩回收,導(dǎo)致內(nèi)存泄露碘橘,考慮下面的例子:

lass?A{?

public?A(){?

?B.getInstance().setA(this);?

?}?

?....?

}?

//B類采用單例模式?

class?B{?

?private?A?a;?

privatestatic?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是個比較復(fù)雜的對象或者集合類型會發(fā)生什么情況识颊。

避免內(nèi)存泄漏的幾點(diǎn)建議:

1祥款、盡早釋放無用對象的引用刃跛。

2桨昙、避免在循環(huán)中創(chuàng)建對象蛙酪。

3桂塞、使用字符串處理時避免使用String,應(yīng)使用StringBuffer。

4吟吝、盡量少使用靜態(tài)變量浙宜,因為靜態(tài)變量存放在永久代同仆,基本不參與垃圾回收俗批。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市干像,隨后出現(xiàn)的幾起案子麻汰,更是在濱河造成了極大的恐慌岔擂,老刑警劉巖位喂,帶你破解...
    沈念sama閱讀 216,591評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件弃舒,死亡現(xiàn)場離奇詭異状原,居然都是意外死亡削锰,警方通過查閱死者的電腦和手機(jī)颅夺,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,448評論 3 392
  • 文/潘曉璐 我一進(jìn)店門唆姐,熙熙樓的掌柜王于貴愁眉苦臉地迎上來赵抢,“玉大人先巴,你說我怎么就攤上這事窄驹。” “怎么了瑞眼?”我有些...
    開封第一講書人閱讀 162,823評論 0 353
  • 文/不壞的土叔 我叫張陵辆影,是天一觀的道長徒像。 經(jīng)常有香客問我锯蛀,道長,這世上最難降的妖魔是什么瞳遍? 我笑而不...
    開封第一講書人閱讀 58,204評論 1 292
  • 正文 為了忘掉前任傅蹂,我火速辦了婚禮,結(jié)果婚禮上份蝴,老公的妹妹穿的比我還像新娘署鸡。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,228評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般塞茅。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上迟蜜,一...
    開封第一講書人閱讀 51,190評論 1 299
  • 那天障斋,我揣著相機(jī)與錄音纵潦,去河邊找鬼。 笑死垃环,一個胖子當(dāng)著我的面吹牛邀层,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播遂庄,決...
    沈念sama閱讀 40,078評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼寥院,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了涛目?” 一聲冷哼從身側(cè)響起秸谢,我...
    開封第一講書人閱讀 38,923評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎霹肝,沒想到半個月后估蹄,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,334評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡沫换,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,550評論 2 333
  • 正文 我和宋清朗相戀三年元媚,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片苗沧。...
    茶點(diǎn)故事閱讀 39,727評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖炭晒,靈堂內(nèi)的尸體忽然破棺而出待逞,到底是詐尸還是另有隱情,我是刑警寧澤网严,帶...
    沈念sama閱讀 35,428評論 5 343
  • 正文 年R本政府宣布识樱,位于F島的核電站,受9級特大地震影響震束,放射性物質(zhì)發(fā)生泄漏怜庸。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,022評論 3 326
  • 文/蒙蒙 一垢村、第九天 我趴在偏房一處隱蔽的房頂上張望割疾。 院中可真熱鬧,春花似錦嘉栓、人聲如沸宏榕。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,672評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽麻昼。三九已至奠支,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間抚芦,已是汗流浹背倍谜。 一陣腳步聲響...
    開封第一講書人閱讀 32,826評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留叉抡,地道東北人尔崔。 一個月前我還...
    沈念sama閱讀 47,734評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像卜壕,于是被迫代替她去往敵國和親您旁。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,619評論 2 354

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