1.1內(nèi)存溢出:(Out Of Memory---OOM)
系統(tǒng)已經(jīng)不能再分配出你所需要的空間,比如你需要100M的空間焊傅,系統(tǒng)只剩90M了,這就叫內(nèi)存溢出
例子:一個盤子用盡各種方法只能裝4個果子鹏倘,你裝了5個薯嗤,結(jié)果掉倒地上不能吃了。這就是溢出骆姐。比方說棧,棧滿時再做進(jìn)棧必定產(chǎn)生空間溢出肉渴,叫上溢,椡妫空時再做退棧也產(chǎn)生空間溢出,稱為下溢券勺。就是分配的內(nèi)存不足以放下數(shù)據(jù)項序列,稱為內(nèi)存溢出。說白了就是我承受不了那么多关炼,那我就報錯,
1.2內(nèi)存泄漏: (Memory Leak)----》強(qiáng)引用所指向的對象不會被回收寸潦,可能導(dǎo)致內(nèi)存泄漏,虛擬機(jī)寧愿拋出OOM也不會去回收他指向的對象
意思就是你用資源的時候為他開辟了一段空間见转,當(dāng)你用完時忘記釋放資源了蒜哀,這時內(nèi)存還被占用著,一次沒關(guān)系凡怎,但是內(nèi)存泄漏次數(shù)多了就會導(dǎo)致內(nèi)存溢出
例子:你向系統(tǒng)申請分配內(nèi)存進(jìn)行使用(new),可是使用完了以后卻不歸還(delete)统倒,結(jié)果你申請到的那塊內(nèi)存你自己也不能再訪問(也許你把它的地址給弄丟了),而系統(tǒng)也不能再次將它分配給需要的程序房匆。就相當(dāng)于你租了個帶鑰匙的柜子,你存完東西之后把柜子鎖上之后井氢,把鑰匙丟了或者沒有將鑰匙還回去,那么結(jié)果就是這個柜子將無法供給任何人使用花竞,也無法被垃圾回收器回收,因為找不到他的任何信息约急。
一般我們所說的內(nèi)存泄漏指的是堆內(nèi)存的泄露苗分,堆內(nèi)存是指程序從堆中分配的,大小隨機(jī)的用完后必須顯示釋放的內(nèi)存摔癣,C++/C中有free函數(shù)可以釋放內(nèi)存纬向,java中有垃圾回收機(jī)制不用程序員自己手動調(diào)用釋放
如果這塊內(nèi)存不釋放戴卜,就不能再用了,這就叫這塊內(nèi)存泄漏了
2.以發(fā)生的方式來分類叉瘩,內(nèi)存泄漏可以分為4類:
- 常發(fā)性內(nèi)存泄漏。發(fā)生內(nèi)存泄漏的代碼會被多次執(zhí)行到薇缅,每次被執(zhí)行的時候都會導(dǎo)致一塊內(nèi)存泄漏。
- 偶發(fā)性內(nèi)存泄漏汤徽。發(fā)生內(nèi)存泄漏的代碼只有在某些特定環(huán)境或操作過程下才會發(fā)生。常發(fā)性和偶發(fā)性是相對的谒府。對于特定的環(huán)境,偶發(fā)性的也許就變成了常發(fā)性的完疫。所以測試環(huán)境和測試方法對檢測內(nèi)存泄漏至關(guān)重要债蓝。
- 一次性內(nèi)存泄漏。發(fā)生內(nèi)存泄漏的代碼只會被執(zhí)行一次饰迹,或者由于算法上的缺陷,導(dǎo)致總會有一塊僅且一塊內(nèi)存發(fā)生泄漏啊鸭。比如,在類的構(gòu)造函數(shù)中分配內(nèi)存赠制,在析構(gòu)函數(shù)中卻沒有釋放該內(nèi)存,所以內(nèi)存泄漏只會發(fā)生一次库正。
- 隱式內(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)存泄漏它更難被檢測到
3.內(nèi)存溢出的原因及解決方法:
1.內(nèi)存溢出原因:
(1)內(nèi)存中加載的數(shù)據(jù)量過于龐大畜普,如一次從數(shù)據(jù)庫取出過多數(shù)據(jù);
(2)集合類中有對對象的引用吃挑,使用完后未清空,使得JVM不能回收舶衬;
(3)代碼中存在死循環(huán)或循環(huán)產(chǎn)生過多重復(fù)的對象實體;
(4)使用的第三方軟件中的BUG逛犹;
(5)啟動參數(shù)內(nèi)存值設(shè)定的過小
2.內(nèi)存溢出的原因及解決方法:
(1)修改JVM啟動參數(shù)梁剔,直接增加內(nèi)存。(-Xms荣病,-Xmx參數(shù)一定不要忘記加渗柿。)
(2)檢查錯誤日志,查看“OutOfMemory”錯誤前是否有其 它異扯淦埽或錯誤。
(3)對代碼進(jìn)行走查和分析混槐,找出可能發(fā)生內(nèi)存溢出的位置。
(4)使用內(nèi)存查看工具動態(tài)查看內(nèi)存使用情況
對代碼分析找出可能發(fā)生內(nèi)存溢出的位置声登, 可能出現(xiàn)的幾種情況:
1、檢查對數(shù)據(jù)庫查詢中件舵,是否有一次獲得全部數(shù)據(jù)的查詢。一般來說铅祸,如果一次取十萬條記錄到內(nèi)存,就可能引起內(nèi)存溢出临梗。這個問題比較隱蔽稼跳,在上線前,數(shù)據(jù)庫中數(shù)據(jù)較少汤善,不容易出問題,上線后红淡,數(shù)據(jù)庫中數(shù)據(jù)多了,一次查詢就有可能引起內(nèi)存溢出在旱。因此對于數(shù)據(jù)庫查詢盡量采用分頁的方式查詢。
2颈渊、檢查代碼中是否有死循環(huán)或遞歸調(diào)用终佛。
3雾家、檢查是否有大循環(huán)重復(fù)產(chǎn)生新對象實體绍豁。
4、檢查List竹揍、MAP等集合對象是否有使用完后,未清除的問題芬位。List、MAP等集合對象會始終存有對對象的引用昧碉,使得這些對象不能被GC回收。