java虛擬機規(guī)范規(guī)定JVM的內(nèi)存分為了好幾塊,比如堆,棧,程序計數(shù)器,方法區(qū)等,而Hotspot jvm的實現(xiàn)中,將堆內(nèi)存分為了三部分,新生代,老年代,持久帶,其中持久帶實現(xiàn)了規(guī)范中規(guī)定的方法區(qū),而內(nèi)存模型中不同的部分都會出現(xiàn)相應(yīng)的OOM錯誤
相信有一定java開發(fā)經(jīng)驗的人或多或少都會遇到OutOfMemoryError的問題,這個問題曾困擾了我很長時間,隨著解決各類問題經(jīng)驗的積累以及對問題根源的探索,終于有了一個比較深入的認識。
在解決java內(nèi)存溢出問題之前,需要對jvm(java虛擬機)的內(nèi)存管理有一定的認識。jvm管理的內(nèi)存大致包括三種不同類型的內(nèi)存區(qū)域:Permanent Generation space(永久保存區(qū)域)、Heap space(堆區(qū)域)、Java Stacks(Java棧)。其中永久保存區(qū)域主要存放Class(類)和Meta的信息,Class第一次被Load的時候被放入PermGen space區(qū)域,Class需要存儲的內(nèi)容主要包括方法和靜態(tài)屬性。堆區(qū)域用來存放Class的實例(即對象),對象需要存儲的內(nèi)容主要是非靜態(tài)屬性。每次用new創(chuàng)建一個對象實例后,對象實例存儲在堆區(qū)域中,這部分空間也被jvm的垃圾回收機制管理。而Java棧跟大多數(shù)編程語言包括匯編語言的棧功能相似,主要基本類型變量以及方法的輸入輸出參數(shù)。Java程序的每個線程中都有一個獨立的堆棧。容易發(fā)生內(nèi)存溢出問題的內(nèi)存空間包括:Permanent Generation space和Heap space。
第一種OutOfMemoryError: PermGen space
JVM管理兩種類型的內(nèi)存,堆和非堆。堆是給開發(fā)人員用的上面說的就是,是在JVM啟動時創(chuàng)建;非堆是留給JVM自己用的,用來存放類的信息的。它和堆不同,運行期內(nèi)GC不會釋放空間。如果web app用了大量的第三方j(luò)ar或者應(yīng)用有太多的class文件而恰好MaxPermSize設(shè)置較小,超出了也會導致這塊內(nèi)存的占用過多造成溢出,或者tomcat熱部署時侯不會清理前面加載的環(huán)境,只會將context更改為新部署的,非堆存的內(nèi)容就會越來越多。
PermGen space的全稱是Permanent Generation space,是指內(nèi)存的永久保存區(qū)域,這塊內(nèi)存主要是被JVM存放Class和Meta信息的,Class在被Loader時就會被放到PermGen space中,它和存放類實例(Instance)的Heap區(qū)域不同,GC(Garbage Collection)不會在主程序運行期對PermGen space進行清理,所以如果你的應(yīng)用中有很CLASS的話,就很可能出現(xiàn)PermGen space錯誤,這種錯誤常見在web服務(wù)器對JSP進行pre compile的時候。如果你的WEB APP下都用了大量的第三方j(luò)ar, 其大小超過了jvm默認的大?。?M)那么就會產(chǎn)生此錯誤信息了。
一個最佳的配置例子:(經(jīng)過本人驗證,自從用此配置之后,再未出現(xiàn)過tomcat死掉的情況)
set JAVA_OPTS=-Xms800m -Xmx800m -XX:PermSize=128M -XX:MaxNewSize=256m -XX:MaxPermSize=256m
發(fā)生這種問題的原意是程序中使用了大量的jar或class,使java虛擬機裝載類的空間不夠,與Permanent Generation space有關(guān)。解決這類問題有以下兩種辦法:
1.增加java虛擬機中的XX:PermSize和XX:MaxPermSize參數(shù)的大小,其中XX:PermSize是初始永久保存區(qū)域大小,XX:MaxPermSize是最大永久保存區(qū)域大小。如針對tomcat6.0,在catalina.sh 或catalina.bat文件中一系列環(huán)境變量名說明結(jié)束處(大約在70行左右) 增加一行: JAVA_OPTS=“ -XX:PermSize=64M -XX:MaxPermSize=128m” 如果是windows服務(wù)器還可以在系統(tǒng)環(huán)境變量中設(shè)置。感覺用tomcat發(fā)布sprint+struts+hibernate架構(gòu)的程序時很容易發(fā)生這種內(nèi)存溢出錯誤。使用上述方法,我成功解決了部署ssh項目的tomcat服務(wù)器經(jīng)常宕機的問題。
2.清理應(yīng)用程序中web-inf/lib下的jar,如果tomcat部署了多個應(yīng)用,很多應(yīng)用都使用了相同的jar,可以將共同的jar移到tomcat共同的lib下,減少類的重復(fù)加載。這種方法是網(wǎng)上部分人推薦的,我沒試過,但感覺減少不了太大的空間,最靠譜的還是第一種方法。
第二種OutOfMemoryError: Java heap space
第一種情況是個補充,主要存在問題就是出現(xiàn)在這個情況中。其默認空間(即-Xms)是物理內(nèi)存的1/64,最大空間(-Xmx)是物理內(nèi)存的1/4。如果內(nèi)存剩余不到40%,JVM就會增大堆到Xmx設(shè)置的值,內(nèi)存剩余超過70%,JVM就會減小堆到Xms設(shè)置的值。所以服務(wù)器的Xmx和Xms設(shè)置一般應(yīng)該設(shè)置相同避免每次GC后都要調(diào)整虛擬機堆的大小。假設(shè)物理內(nèi)存無限大,那么JVM內(nèi)存的最大值跟操作系統(tǒng)有關(guān),一般32位機是1.5g到3g之間,而64位的就不會有限制了。
注意:如果Xms超過了Xmx值,或者堆最大值和非堆最大值的總和超過了物理內(nèi)存或者操作系統(tǒng)的最大限制都會引起服務(wù)器啟動不起來。
發(fā)生這種問題的原因是java虛擬機創(chuàng)建的對象太多,在進行垃圾回收之間,虛擬機分配的到堆內(nèi)存空間已經(jīng)用滿了,與Heap space有關(guān)。解決這類問題有兩種思路:
1.檢查程序,看是否有死循環(huán)或不必要地重復(fù)創(chuàng)建大量對象。找到原因后,修改程序和算法。 我以前寫一個使用K-Means文本聚類算法對幾萬條文本記錄(每條記錄的特征向量大約10來個)進行文本聚類時,由于程序細節(jié)上有問題,就導致了Java heap space的內(nèi)存溢出問題,后來通過修改程序得到了解決。
2.增加Java虛擬機中Xms(初始堆大?。┖蚗mx(最大堆大?。﹨?shù)的大小。如:set JAVA_OPTS= -Xms256m -Xmx1024m
第三種OutOfMemoryError:unable to create new native thread
在java應(yīng)用中,有時候會出現(xiàn)這樣的錯誤:OutOfMemoryError: unable to create new native thread.這種怪事是因為JVM已經(jīng)被系統(tǒng)分配了大量的內(nèi)存(比如1.5G),并且它至少要占用可用內(nèi)存的一半。有人發(fā)現(xiàn),在線程個數(shù)很多的情況下,你分配給JVM的內(nèi)存越多,那么,上述錯誤發(fā)生的可能性就越大。
那么是什么原因造成這種問題呢?
每一個32位的進程最多可以使用2G的可用內(nèi)存,因為另外2G被操作系統(tǒng)保留。這里假設(shè)使用1.5G給JVM,那么還余下500M可用內(nèi)存。這500M內(nèi)存中的一部分必須用于系統(tǒng)dll的加載,那么真正剩下的也許只有400M,現(xiàn)在關(guān)鍵的地方出現(xiàn)了:當你使用Java創(chuàng)建一個線程,在JVM的內(nèi)存里也會創(chuàng)建一個Thread對象,但是同時也會在操作系統(tǒng)里創(chuàng)建一個真正的物理線程(參考JVM規(guī)范),操作系統(tǒng)會在余下的400兆內(nèi)存里創(chuàng)建這個物理線程,而不是在JVM的1500M的內(nèi)存堆里創(chuàng)建。在jdk1.4里頭,默認的棧大小是256KB,但是在jdk1.5里頭,默認的棧大小為1M每線程,因此,在余下400M的可用內(nèi)存里邊我們最多也只能創(chuàng)建400個可用線程。
這樣結(jié)論就出來了,要想創(chuàng)建更多的線程,你必須減少分配給JVM的最大內(nèi)存。還有一種做法是讓JVM宿主在你的JNI代碼里邊。
給出一個有關(guān)能夠創(chuàng)建線程的最大個數(shù)的估算公式:
?。∕axProcessMemory - JVMMemory - ReservedOsMemory) / (ThreadStackSize) = Number of threads
對于jdk1.5而言,假設(shè)操作系統(tǒng)保留120M內(nèi)存:
1.5GB JVM: (2GB-1.5Gb-120MB)/(1MB) = ~380 threads 1.0GB JVM: (2GB-1.0Gb-120MB)/(1MB) = ~880 threads
對于棧大小為256KB的jdk1.4而言,
1.5GB allocated to JVM: ~1520 threads 1.0GB allocated to JVM: ~3520 threads
對于這個異常我們首先需要判斷下,發(fā)生內(nèi)存溢出時進程中到底都有什么樣的線程,這些線程是否是應(yīng)該存在的,是否可以通過優(yōu)化來降低線程數(shù); 另外一方面默認情況下java為每個線程分配的棧內(nèi)存大小是1M,通常情況下,這1M的棧內(nèi)存空間是足足夠用了,因為在通常在棧上存放的只是基礎(chǔ)類型的數(shù)據(jù)或者對象的引用,這些東西都不會占據(jù)太大的內(nèi)存, 我們可以通過調(diào)整jvm參數(shù),降低為每個線程分配的棧內(nèi)存大小來解決問題,例如在jvm參數(shù)中添加-Xss128k將線程棧內(nèi)存大小設(shè)置為128k。
垃圾回收GC的角色
JVM調(diào)用GC的頻度還是很高的,主要兩種情況下進行垃圾回收:
當應(yīng)用程序線程空閑;另一個是java內(nèi)存堆不足時,會不斷調(diào)用GC,若連續(xù)回收都解決不了內(nèi)存堆不足的問題時,就會報out of memory錯誤。因為這個異常根據(jù)系統(tǒng)運行環(huán)境決定,所以無法預(yù)期它何時出現(xiàn)。
根據(jù)GC的機制,程序的運行會引起系統(tǒng)運行環(huán)境的變化,增加GC的觸發(fā)機會。
為了避免這些問題,程序的設(shè)計和編寫就應(yīng)避免垃圾對象的內(nèi)存占用和GC的開銷。顯示調(diào)用System.GC()只能建議JVM需要在內(nèi)存中對垃圾對象進行回收,但不是必須馬上回收,
一個是并不能解決內(nèi)存資源耗空的局面,另外也會增加GC的消耗
評論
查看更多