一、內(nèi)存溢出的定義和原因
定義
內(nèi)存溢出是指應(yīng)用系統(tǒng)中存在無法回收的內(nèi)存或使用的內(nèi)存過多,,最終使得程序運行要用到的內(nèi)存大于虛擬機能提供的最大內(nèi)存,。為了解決Java中內(nèi)存溢出問題,我們首先必須了解Java是如何管理內(nèi)存的,。Java的內(nèi)存管理就是對象的分配和釋放問題,。在Java中,內(nèi)存的分配是由程序完成的,,而內(nèi)存的釋放是由垃圾收集器(GarbageCollection,,GC)完成的,程序員不需要通過調(diào)用GC函數(shù)來釋放內(nèi)存,,因為不同的JVM實現(xiàn)者可能使用不同的算法管理GC,,有的是內(nèi)存使用到達(dá)一定程度時,GC才開始工作,,也有定時執(zhí)行的,,有的是中斷式執(zhí)行GC。但GC只能回收無用并且不再被其它對象引用的那些對象所占用的空間,。Java的內(nèi)存垃圾回收機制是從程序的主要運行對象開始檢查引用鏈,,當(dāng)遍歷一遍后發(fā)現(xiàn)沒有被引用的孤立對象就作為垃圾回收,。
原因
1、內(nèi)存中加載的數(shù)據(jù)量過于龐大,,如一次從數(shù)據(jù)庫取出過多數(shù)據(jù),。
2、集合類中有對對象的引用,,使用完后未清空,,使得JVM不能回收。
3,、代碼中存在死循環(huán)或循環(huán)產(chǎn)生過多重復(fù)的對象實體,。
4、使用的第三方軟件中的BUG,。
5,、啟動參數(shù)設(shè)定的過小。
二,、內(nèi)存溢出的解決問題
第一步,,就是修改JVM啟動參數(shù),直接增加內(nèi)存,。這一點看上去似乎很簡單,,但很容易被忽略。JVM默認(rèn)可以使用的內(nèi)存為64M,,Tomcat默認(rèn)可以使用的內(nèi)存為128MB,,對于稍復(fù)雜一點的系統(tǒng)就會不夠用。在某項目中,,就因為啟動參數(shù)使用的默認(rèn)值,,經(jīng)常報“OutOfMemory”錯誤。因此,,-Xms,,-Xmx參數(shù)一定不要忘記加。
第二步,,檢查錯誤日志,,查看“OutOfMemory”錯誤前是否有其它異常或錯誤,。在一個項目中,,使用兩個數(shù)據(jù)庫連接,其中專用于發(fā)送短信的數(shù)據(jù)庫連接使用DBCP連接池管理,,用戶為不將短信發(fā)出,,有意將數(shù)據(jù)庫連接用戶名改錯,使得日志中有許多數(shù)據(jù)庫連接異常的日志,一段時間后,,就出現(xiàn)“OutOfMemory”錯誤,。經(jīng)分析,這是由于DBCP連接池BUG引起的,,數(shù)據(jù)庫連接不上后,,沒有將連接釋放,最終使得DBCP報“OutOfMemory”錯誤,。經(jīng)過修改正確數(shù)據(jù)庫連接參數(shù)后,,就沒有再出現(xiàn)內(nèi)存溢出的錯誤。
查看日志對于分析內(nèi)存溢出是非常重要的,,通過仔細(xì)查看日志,,分析內(nèi)存溢出前做過哪些操作,可以大致定位有問題的模塊,。
第三步,安排有經(jīng)驗的編程人員對代碼進(jìn)行走查和分析,,找出可能發(fā)生內(nèi)存溢出的位置,。重點排查以下幾點:
1、檢查代碼中是否有死循環(huán)或遞歸調(diào)用,。
2,、檢查是否有大循環(huán)重復(fù)產(chǎn)生新對象實體。
3,、檢查對數(shù)據(jù)庫查詢中,,是否有一次獲得全部數(shù)據(jù)的查詢。一般來說,,如果一次取十萬條記錄到內(nèi)存,,就可能引起內(nèi)存溢出。這個問題比較隱蔽,,在上線前,,數(shù)據(jù)庫中數(shù)據(jù)較少,不容易出問題,,上線后,,數(shù)據(jù)庫中數(shù)據(jù)多了,一次查詢就有可能引起內(nèi)存溢出,。因此對于數(shù)據(jù)庫查詢盡量采用分頁的方式查詢,。
4、檢查List,、MAP等集合對象是否有使用完后,,未清除的問題。List、MAP等集合對象會始終存有對對象的引用,,使得這些對象不能被GC回收,。
第四步,使用內(nèi)存查看工具動態(tài)查看內(nèi)存使用情況,。某個項目上線后,,每次系統(tǒng)啟動兩天后,就會出現(xiàn)內(nèi)存溢出的錯誤,。這種情況一般是代碼中出現(xiàn)了緩慢的內(nèi)存泄漏,,用上面三個步驟解決不了,這就需要使用內(nèi)存查看工具了,。
內(nèi)存查看工具有許多,,比較有名的有:Optimizeit Profiler、JProbeProfiler,、JinSight和Java1.5的Jconsole等,。它們的基本工作原理大同小異,都是監(jiān)測Java程序運行時所有對象的申請,、釋放等動作,,將內(nèi)存管理的所有信息進(jìn)行統(tǒng)計、分析,、可視化,。開發(fā)人員可以根據(jù)這些信息判斷程序是否有內(nèi)存泄漏問題。一般來說,,一個正常的系統(tǒng)在其啟動完成后其內(nèi)存的占用量是基本穩(wěn)定的,,而不應(yīng)該是無限制的增長的。持續(xù)地觀察系統(tǒng)運行時使用的內(nèi)存的大小,,可以看到在內(nèi)存使用監(jiān)控窗口中是基本規(guī)則的鋸齒形的圖線,,如果內(nèi)存的大小持續(xù)地增長,則說明系統(tǒng)存在內(nèi)存泄漏問題,。通過間隔一段時間取一次內(nèi)存快照,,然后對內(nèi)存快照中對象的使用與引用等信息進(jìn)行比對與分析,可以找出是哪個類的對象在泄漏,。
通過以上四個步驟的分析與處理,,基本能處理內(nèi)存溢出的問題。當(dāng)然,,在這些過程中也需要相當(dāng)?shù)慕?jīng)驗與敏感度,,需要在實際的開發(fā)與調(diào)試過程中不斷積累。
聲明:以上方法源于程序系統(tǒng)索引或網(wǎng)民分享提供,,僅供您參考使用,,不代表本網(wǎng)站的研究觀點,證明有效,請注意甄別內(nèi)容來源的真實性和權(quán)威性,。申請刪除>> 糾錯>>