基本概念
內存溢出 out of memory,是指程序在申請內存時,沒有足夠的內存空間供其使用,出現out of memory;比如系統只有存放integer的空間,但你卻申請了存放long,那就是內存溢出。比方說,定義了20個字節大小的內存空間,卻寫入了21個字節的數據。通俗的說,就是內存不夠,沒辦法支持當前程序。(當發生內存溢出時,程序將無法進行,強制終止。)
內存泄露 memory leak,是指程序在申請內存后,無法釋放已申請的內存空間,一次內存泄露危害可以忽略,但內存泄露堆積后果很嚴重,無論多少內存,遲早會被占光。(如果發生內存泄露,那么可用內存會逐漸減少,從而降低性能。)
memory leak會最終會導致out of memory!
內存溢出就是你要求分配的內存超出了系統能給你的,系統不能滿足需求,于是產生溢出。
內存泄漏是指你向系統申請分配內存進行使用(new),可是使用完了以后卻不歸還(delete),結果你申請到的那塊內存你自己也不能再訪問(也許你把它的地址給弄丟了),而系統也不能再次將它分配給需要的程序。一個盤子用盡各種方法只能裝4個果子,你裝了5個,結果掉倒地上不能吃了。這就是溢出!比方說棧,棧滿時再做進棧必定產生空間溢出,叫上溢,棧空時再做退棧也產生空間溢出,稱為下溢。就是分配的內存不足以放下數據項序列,稱為內存溢出.
java可能出現的內存溢出
1、在程序中存在死循環,或者循環過多,而產生了過多重復的對象的實例;
2、存在對象的引用,使用完后沒有清除,導致JAVA虛擬機不能回收;
3、一次操 作時,在內存中加載了大量的數據;
原則上來說,在JAVA中,由于它的自動垃圾回收機制,出現內存溢出的可能性并不是很大【會出現】。
內存泄露出現類型
以發生的方式來分類,內存泄漏可以分為4類:
1. 常發性內存泄漏。發生內存泄漏的代碼會被多次執行到,每次被執行的時候都會導致一塊內存泄漏。
2. 偶發性內存泄漏。發生內存泄漏的代碼只有在某些特定環境或操作過程下才會發生。常發性和偶發性是相對的。對于特定的環境,偶發性的也許就變成了常發性的。所以測試環境和測試方法對檢測內存泄漏至關重要。
3. 一次性內存泄漏。發生內存泄漏的代碼只會被執行一次,或者由于算法上的缺陷,導致總會有一塊僅且一塊內存發生泄漏。比如,在類的構造函數中分配內存,在析構函數中卻沒有釋放該內存,所以內存泄漏只會發生一次。
4. 隱式內存泄漏。程序在運行過程中不停的分配內存,但是直到結束的時候才釋放內存。嚴格的說這里并沒有發生內存泄漏,因為最終程序釋放了所有申請的內存。但是對于一個服務器程序,需要運行幾天,幾周甚至幾個月,不及時釋放內存也可能導致最終耗盡系統的所有內存。所以,我們稱這類內存泄漏為隱式內存泄漏。
內存泄露的后果
內存泄漏會因為減少可用內存的數量從而降低計算機的性能。最終,在最糟糕的情況下,過多的可用內存被分配掉導致全部或部分設備停止正常工作,或者應用程序崩潰。
內存泄漏可能不嚴重,甚至能夠被常規的手段檢測出來。在現代操作系統中,一個應用程序使用的常規內存在程序終止時被釋放。這表示一個短暫運行的應用程序中的內存泄漏不會導致嚴重后果。
在以下情況,內存泄漏導致較嚴重的后果:
*程序運行后置之不理,并且隨著時間的流失消耗越來越多的內存(比如服務器上的后臺任務,尤其是嵌入式系統中的后臺任務,這些任務可能被運行后很多年內都置之不理);
* 新的內存被頻繁地分配,比如當顯示電腦游戲或動畫視頻畫面時;
* 程序能夠請求未被釋放的內存(比如共享內存),甚至是在程序終止的時候;
* 泄漏在操作系統內部發生;
* 泄漏在系統關鍵驅動中發生;
* 內存非常有限,比如在嵌入式系統或便攜設備中;
* 當運行于一個終止時內存并不自動釋放的操作系統(比如AmigaOS)之上,而且一旦丟失只能通過重啟來恢復。
總結
從用戶使用程序的角度來看,內存泄漏本身不會產生什么危害,作為一般的用戶,根本感覺不到內存泄漏的存在。真正有危害的是內存泄漏的堆積,這會最終消耗盡系統所有的內存。從這個角度來說,一次性內存泄漏并沒有什么危害,因為它不會堆積,而隱式內存泄漏危害性則非常大,因為較之于常發性和偶發性內存泄漏它更難被檢測到 。
JAVA的垃圾回收機制彰顯了JAVA的健壯性與安全性,合理的設計代碼,有效的評估內存使用情況,基本上不會出現上述問題。
致謝:感謝您的耐心閱讀!
新聞熱點
疑難解答