內存溢出和內存泄漏


內存溢出 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)即會引起。 

Java代碼  
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對象都會獨立占用內存一塊區域,如: 

Java代碼  
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


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM