內存溢出 out of memory,是指程序在申請內存時,沒有足夠的內存空間供其使用,出現out of memory;比如申請了一個integer,但給它存了long才能存下的數,那就是內存溢出。
內存泄露 memory leak,是指程序在申請內存后,無法釋放已申請的內存空間,一次內存泄露危害可以忽略,但內存泄露堆積后果很嚴重,無論多少內存,遲早會被占光。
memory leak會最終會導致out of memory!
內存溢出就是你要求分配的內存超出了系統能給你的,系統不能滿足需求,於是產生溢出。
內存泄漏是指你向系統申請分配內存進行使用(new),可是使用完了以后卻不歸還(delete),結果你申請到的那塊內存你自己也不能再訪問(也許你把它的地址給弄丟了),而系統也不能再次將它分配給需要的程序。一個盤子用盡各種方法只能裝4個果子,你裝了5個,結果掉倒地上不能吃了。這就是溢出!比方說棧,棧滿時再做進棧必定產生空間溢出,叫上溢,棧空時再做退棧也產生空間溢出,稱為下溢。就是分配的內存不足以放下數據項序列,稱為內存溢出.
以發生的方式來分類,內存泄漏可以分為4類:
1. 常發性內存泄漏。發生內存泄漏的代碼會被多次執行到,每次被執行的時候都會導致一塊內存泄漏。
2. 偶發性內存泄漏。發生內存泄漏的代碼只有在某些特定環境或操作過程下才會發生。常發性和偶發性是相對的。對於特定的環境,偶發性的也許就變成了常發性的。所以測試環境和測試方法對檢測內存泄漏至關重要。
3. 一次性內存泄漏。發生內存泄漏的代碼只會被執行一次,或者由於算法上的缺陷,導致總會有一塊僅且一塊內存發生泄漏。比如,在類的構造函數中分配內存,在析構函數中卻沒有釋放該內存,所以內存泄漏只會發生一次。
4. 隱式內存泄漏。程序在運行過程中不停的分配內存,但是直到結束的時候才釋放內存。嚴格的說這里並沒有發生內存泄漏,因為最終程序釋放了所有申請的內存。但是對於一個服務器程序,需要運行幾天,幾周甚至幾個月,不及時釋放內存也可能導致最終耗盡系統的所有內存。所以,我們稱這類內存泄漏為隱式內存泄漏。
從用戶使用程序的角度來看,內存泄漏本身不會產生什么危害,作為一般的用戶,根本感覺不到內存泄漏的存在。真正有危害的是內存泄漏的堆積,這會最終消耗盡系統所有的內存。從這個角度來說,一次性內存泄漏並沒有什么危害,因為它不會堆積,而隱式內存泄漏危害性則非常大,因為較之於常發性和偶發性內存泄漏它更難被檢測到
相關問題
1. Q:Java中會存在內存泄漏嗎?
A: 當然也會的。當被分配的對象可達但已無用(未對作廢數據內存單元的引用置null)即會引起。
Vector v=new Vector(10); for (int i=1;i<100; i) { Object o=new Object(); v.add(o); o=null; } // 此時,所有的Object對象都沒有被釋放,因為變量v引用這些對象。 // 對象加入到Vector后,還必須從Vector中刪除,最簡單釋放方法就是將Vector對象設置為null。
2. Q:內存泄露、溢出的異同?
A: 同:都會導致應用程序運行出現問題,性能下降或掛起。
異:
1) 內存泄露是導致內存溢出的原因之一;內存泄露積累起來將導致內存溢出。
2) 內存泄露可以通過完善代碼來避免;內存溢出可以通過調整配置來減少發生頻率,但無法徹底避免。
3. 如何檢測內存泄露?
A: 可以通過一些性能監測分析工具,如 JProfiler、OptimizeitProfiler。
4. Q: 如何避免內存泄露、溢出?
A: 1)盡早釋放無用對象的引用。
好的辦法是使用臨時變量的時候,讓引用變量在退出活動域后自動設置為null,暗示垃圾收集器來收集該對象,防止發生內存泄露。
2)程序進行字符串處理時,盡量避免使用String,而應使用StringBuffer。
因為每一個String對象都會獨立占用內存一塊區域,如:
String str = "aaa"; String str2 = "bbb"; String str3 = str str2; // 假如執行此次之后str , str2再不被調用,那么它們就會在內存中等待GC回收; // 假如程序中存在過多的類似情況就會出現內存錯誤;
3) 盡量少用靜態變量。
因為靜態變量是全局的,GC不會回收。
4)避免集中創建對象尤其是大對象,如果可以的話盡量使用流操作。
JVM會突然需要大量內存,這時會觸發GC優化系統內存環境; 一個案例如下:
5)盡量運用對象池技術以提高系統性能。
生命周期長的對象擁有生命周期短的對象時容易引發內存泄漏,例如大集合對象擁有大數據量的業務對象的時候,可以考慮分塊進行處理,然后解決一塊釋放一塊的策略。
6)不要在經常調用的方法中創建對象,尤其是忌諱在循環中創建對象。
可以適當的使用hashtable,vector創建一組對象容器,然后從容器中去取那些對象,而不用每次new之后又丟棄。
7) 優化配置。
5. Q:內存溢出的解決方案?
A: 一是從代碼層面進行優化完善,盡量避免該情況發生;
二是調整優化服務器配置:
1) 設置-Xms、-Xmx相等;
2) 設置NewSize、MaxNewSize相等;
3) 設置Heap size, PermGen space:
Tomcat 的配置示例:修改 %TOMCAT_HOME%/bin/catalina.bat or catalina.sh
JAVA_OPTS=-Xms800m -Xmx800m -XX:PermSize=128M -XX:MaxNewSize=256m -XX:MaxPermSize=256m
轉自http://825635381.iteye.com/blog/2338162
https://www.cnblogs.com/theWayToAce/p/7759923.html
