內存泄漏與內存溢出

內存泄露

內存泄漏memory leak 是指那些本應該回收(再也不使用)的內存對象沒法被系統回收的現象。在c++中須要程序猿手動釋放內存對象,因此在C++中更容易存在內存泄漏。java引入了自動回收機制,使得在C++中使人頭疼的內存問題獲得了有效的改善,但這並不意味着java程序員不關注內存,由於垃圾回收機制不能徹底保證內存對象在該釋放的地方釋放,現代java虛擬機中廣泛使用根集算法去計算對象的引用可達性,不可達的才能回收,例以下圖中的無用對象被有用對象引用着,致使無用對象引用一直可達,系統回收器不敢冒然回收,從而形成內存泄漏。若是發生內存泄露,那麼可用內存會逐漸減小,從而下降性能。java



內存溢出

內存溢出out of memory,是指程序在爲自身申請內存時,沒有足夠的內存空間供本身使用,出現out of memory;好比你爲程序申請了一個integer,可是隻給它存了long才能存下的數,就是內存溢出。內存溢出就是你要求被分配的內存超出了系統能給你的內存,系統不能知足你的需求,因而產生溢出。當發生內存溢出時,程序將沒法進行,強制終止c++

兩者關係

memory leak最終會致使out of memory!程序員

Java可能出現的內存溢出

  • 在程序中存在死循環,或者循環過多,而產生了過多重複的對象的實例;
  • 存在對象的引用,使用完後沒有清除,致使JAVA虛擬機不能回收;
  • 一次操做時,在內存中加載了大量的數據;

  原則上來講,在JAVA中,因爲它的自動垃圾回收機制,出現內存溢出的可能性並非很大,但仍是會出現。算法

內存泄露出現類型

   以發生的方式來分類,內存泄漏能夠分爲4類: 安全

  • 常發性內存泄漏。發生內存泄漏的代碼會被屢次執行到,每次被執行的時候都會致使一塊內存泄漏。 
  • 偶發性內存泄漏。發生內存泄漏的代碼只有在某些特定環境或操做過程下才會發生。常發性和偶發性是相對的。對於特定的環境,偶發性的也許就變成了常發性的。因此測試環境和測試方法對檢測內存泄漏相當重要。 
  • 一次性內存泄漏。發生內存泄漏的代碼只會被執行一次,或者因爲算法上的缺陷,致使總會有一塊僅且一塊內存發生泄漏。好比,在類的構造函數中分配內存,在析構函數中卻沒有釋放該內存,因此內存泄漏只會發生一次。 
  • 隱式內存泄漏。程序在運行過程當中不停的分配內存,可是直到結束的時候才釋放內存。嚴格的說這裏並無發生內存泄漏,由於最終程序釋放了全部申請的內存。但 是對於一個服務器程序,須要運行幾天,幾周甚至幾個月,不及時釋放內存也可能致使最終耗盡系統的全部內存。因此,咱們稱這類內存泄漏爲隱式內存泄漏。 

內存泄露的後果

    內存泄漏會由於減小可用內存的數量從而下降計算機的性能。最終,在最糟糕的狀況下,過多的可用內存被分配掉致使所有或部分設備中止正常工做,或者應用程序崩潰。
  內存泄漏可能不嚴重,甚至可以被常規的手段檢測出來。在現代操做系統中,一個應用程序使用的常規內存在程序終止時被釋放。這表示一個短暫運行的應用程序中的內存泄漏不會致使嚴重後果。
  在如下狀況,內存泄漏致使較嚴重的後果服務器

  •  程序運行後置之不理,而且隨着時間的流失消耗愈來愈多的內存(好比服務器上的後臺任務,尤爲是嵌入式系統中的後臺任務,這些任務可能被運行後不少年內都置之不理);
  •  新的內存被頻繁地分配,好比當顯示電腦遊戲或動畫視頻畫面時;
  •  程序可以請求未被釋放的內存(好比共享內存),甚至是在程序終止的時候;
  •  泄漏在操做系統內部發生;
  •  泄漏在系統關鍵驅動中發生;
  •  內存很是有限,好比在嵌入式系統或便攜設備中;

  當運行於一個終止時內存並不自動釋放的操做系統(好比AmigaOS)之上,並且一旦丟失只能經過重啓來恢復。函數

 

  從用戶使用程序的角度來看,內存泄漏自己不會產生什麼危害,做爲通常的用戶,根本感受不到內存泄漏的存在。真正有危害的是內存泄漏的堆積,這會最終消耗盡系 統全部的內存。從這個角度來講,一次性內存泄漏並無什麼危害,由於它不會堆積,而隱式內存泄漏危害性則很是大,由於較之於常發性和偶發性內存泄漏它更難 被檢測到 。性能

  JAVA的垃圾回收機制彰顯了JAVA的健壯性與安全性,合理的設計代碼,有效的評估內存使用狀況,基本上不會出現上述問題。測試

相關文章
相關標籤/搜索