1. 程式人生 > >java.lang.OutOfMemoryError: Java heap space記憶體不足問題

java.lang.OutOfMemoryError: Java heap space記憶體不足問題

問題描述
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space

解決方案[轉]
一直都知道可以設定jvm heap大小,一直用eclipse寫/除錯java程式。一直用命令列or console加引數跑程式。現象:在eclipse的配置檔案eclipse.ini中設定-vmargs -Xms500m -Xmx1024m, 在eclipse中直接run 或者debug某些耗記憶體的程式時依然出現java.lang.OutOfMemoryError: Java Heap Space錯誤,即通常認為的記憶體不足,java虛擬機器記憶體不夠用。而在命令列加這些引數則有效果,不會出錯。這說明一個問題,這些引數根本沒有起作用。 今天需要在eclipse裡除錯程式,還沒到需要除錯的地方就heap error了,在網上搜了很多地方,得到了最終的答案:
選中被執行的類,點選選單‘run->run...

’,選擇(x)=Argument標籤頁下的vm arguments框裡
輸入 -Xmx800m, 儲存執行。
原來還需要對每個project單獨設定,汗...


有三種可能導致OutOfMemoryError。

首先是,此JVM有真實的記憶體洩漏,導致此JVM堆在內部實現時產生了一個Bug。這極不可靠。所有JVM都經過充分的測試,並且,如果有人發現這種bug,它將絕對是最高的優先順序。因此你可以非常寬心地排除這種可能性。

第二種可能的OutOfMemoryError原因只不過是,你沒有為你的應用程式執行時給予足夠多的可用記憶體。這種情況,有兩種可能的方案,或者增加 JVM堆可用大小,或者減少你的應用程式所需的記憶體總量。提高JVM可用堆大小可以簡單的使用JVM的 -Xmx 引數。假如你將此引數設定儘可能的大(可用記憶體極限不要超過系統實體記憶體,否則你的應用程式將分頁並暫停),仍然有以上所提到的記憶體問題,那麼,你需要減 少你的應用程式所可能用到記憶體總量。減少應用程式記憶體可能是簡單的,你可能允許一些集合過大,例如使用了許多大的緩衝區。或者它過於複雜,要求你重新實現 一些類,乃至重新設計應用程式。

讀者 Jams Stauffer 指出有些JVM(例如 sun的 JVMs),還有一個“Perm”引數用來處理JVM結構與類物件。如果你正在使用一個數量非常巨大的類集,它有可能執行在"Perm"空間之外,然後你 需要增加此空間的大小,例如,sun的JVM使用 -XX:PermSize 與 -XX:MaxPermSize 選項。

第三種導致OutOfMemoryError最為常見,無心的物件引用保持。你沒有明確無誤的釋放物件,以致於你的堆增長再增長,直到你沒有額外的空間。

處理OutOfMemoryError:

是JVM內部的BUG?不太可能。如果是,這是優先順序最高的BUG(為什麼還沒有人發現它,而你碰到了?)。

沒有足夠的記憶體分配給實際執行的應用程式?兩種選擇:使用-Xmx引數增加堆的最大使用記憶體(或者使用-XX:MaxPermSize引數增加Perm空 間大小); 或者使用更小的集合/緩衝區/表空間/物件.....,以減少所需要的記憶體總量,也就是說,可以調整物件大小,重新設計與重新實現你的應用程 序。

無心的物件引用保持?找到保持這些無意引用的源物件,改變它並釋放這些物件。在IBM開發者社群的文章綱要式的揭示了這樣一個通用的處理過程。這個過程主 要是等到應用程式到達恆定狀態--你將期望最多的新建立的物件是臨時物件,並且可以被垃圾收集器收集。這常常是在應用程式所有的初始化工作完成之後。

強迫垃圾收集,獲得一個堆的物件快照。
做任何工作可能正在導到無意的物件引用保持。
強迫另一次垃圾收集並獲得第二次堆的物件快照。
比較這兩個快照,觀察從第一個快照到第二個快照哪些物件在數量上有所增加。因為你在快照之前強迫垃圾收集,剩下的將是所有被應用程式引用的物件,比較兩個快照將準確的標識那些新建立的、保留在應用程式裡的物件。
根據你對應用程式的認識,決定兩個快照比較中,哪些物件正在無意的保持物件引用。
跟蹤前導引用,找到哪些物件正在引用這些無意的保持物件,直到你找到導致此問題的源物件

啟動虛擬機器的時候,加上一個引數:-Xms800m -Xmx800m就好了 
-Xms <size> 
設定JVM初始化堆記憶體大小 

-Xmx <size> 
設定JVM最大的堆記憶體大小 


如果是應用程式,則:java -Xms800m -Xmx800m 你的類名 
如果是tomcat之類的web伺服器,在這個伺服器的啟動檔案後面加上這個引數即可。


另外設定環境變數
JAVA_OPTS="-server -Xms800m -Xmx800m -XX:PermSize=64M -XX:MaxNewSize=256m -XX:MaxPermSize=128m -Djava.awt.headless=true "

//轉2

//首先檢查程式有沒有限入死迴圈

這個問題主要還是由這個問題 java.lang.OutOfMemoryError: Java heap space 引起的。第一次出現這樣的的問題以後,引發了其他的問題。在網上一查可能是JAVA的堆疊設定太小的原因。
跟據網上的答案大致有這兩種解決方法:
1、設定環境變數

解決方法:手動設定Heap size
修改TOMCAT_HOME/bin/catalina.sh
set JAVA_OPTS= -Xms32m -Xmx512m
可以根據自己機器的記憶體進行更改。

2、java -Xms32m -Xmx800m className
就是在執行JAVA類檔案時加上這個引數,其中className是需要執行的確類名。(包括包名)
這個解決問題了。而且執行的速度比沒有設定的時候快很多。

如果在測試的時候可能會用Eclispe 這時候就需要在Eclipse ->run -arguments 中的VM arguments 中輸入-Xms32m -Xmx800m這個引數就可以了。

後來在Eclilpse中修改了啟動引數,在VM arguments 加入了-Xms32m -Xmx800m,問題解決。

一、java.lang.OutOfMemoryError: PermGen space

PermGen space的全稱是Permanent Generation space,是指記憶體的永久儲存區域,
這塊記憶體主要是被JVM存放Class和Meta資訊的,Class在被Loader時就會被放到PermGen space中,
它和存放類例項(Instance)的Heap區域不同,GC(Garbage Collection)不會在主程式執行期對
PermGen space進行清理,所以如果你的應用中有很多CLASS的話,就很可能出現PermGen space錯誤,
這種錯誤常見在web伺服器對JSP進行pre compile的時候。如果你的WEB APP下都用了大量的第三方jar, 其大小
超過了jvm預設的大小(4M)那麼就會產生此錯誤資訊了。
解決方法: 手動設定MaxPermSize大小

修改TOMCAT_HOME/bin/catalina.sh
在“echo "Using CATALINA_BASE:   $CATALINA_BASE"”上面加入以下行:
JAVA_OPTS="-server -XX:PermSize=64M -XX:MaxPermSize=128m
建議:將相同的第三方jar檔案移置到tomcat/shared/lib目錄下,這樣可以達到減少jar 文件重複佔用記憶體的目的。

二、java.lang.OutOfMemoryError: Java heap space
Heap size 設定
JVM堆的設定是指java程式執行過程中JVM可以調配使用的記憶體空間的設定.JVM在啟動的時候會自動設定Heap size的值,
其初始空間(即-Xms)是實體記憶體的1/64,最大空間(-Xmx)是實體記憶體的1/4。可以利用JVM提供的-Xmn -Xms -Xmx等選項可
進行設定。Heap size 的大小是Young Generation 和Tenured Generaion 之和。
提示:在JVM中如果98%的時間是用於GC且可用的Heap size 不足2%的時候將丟擲此異常資訊。
提示:Heap Size 最大不要超過可用實體記憶體的80%,一般的要將-Xms和-Xmx選項設定為相同,而-Xmn為1/4的-Xmx值。 
解決方法:手動設定Heap size
修改TOMCAT_HOME/bin/catalina.sh
在“echo "Using CATALINA_BASE:   $CATALINA_BASE"”上面加入以下行:
JAVA_OPTS="-server -Xms800m -Xmx800m   -XX:MaxNewSize=256m"

三、例項,以下給出1G記憶體環境下java jvm 的引數設定參考:

JAVA_OPTS="-server -Xms800m -Xmx800m -XX:PermSize=64M -XX:MaxNewSize=256m -XX:MaxPermSize=128m -Djava.awt.headless=true "

很大的web工程,用tomcat預設分配的記憶體空間無法啟動,如果不是在myeclipse中啟動tomcat可以對tomcat這樣設定:

TOMCAT_HOME/bin/catalina.bat 中新增這樣一句話:

          set JAVA_OPTS= -Xmx1024M -Xms512M -XX:MaxPermSize=256m

如果要在myeclipse中啟動,上述的修改就不起作用了,可如下設定:

Myeclipse->preferences->myeclipse->servers->tomcat->tomcat×.×->JDK面板中的

Optional Java VM arguments中新增:-Xmx1024M -Xms512M -XX:MaxPermSize=256m

Optional Java VM arguments中新增:-Xmx1024M -Xms512M -XX:MaxPermSize=256m

挺靈的。