相信有必定java開發經驗的人或多或少都會遇到OutOfMemoryError的問題,這個問題曾困擾了我很長時間,隨着解決各種問題經驗的積累以及對問題根源的探索,終於有了一個比較深刻的認識。java
在解決java內存溢出問題以前,須要對jvm(java虛擬機)的內存管理有必定的認識。jvm管理的內存大體包括三種不一樣類型的內存區域:Permanent Generation space(永久保存區域)、Heap space(堆區域)、Java Stacks(Java棧)。其中永久保存區域主要存放Class(類)和Meta的信息,Class第一次被Load的時候被放入PermGen space區域,Class須要存儲的內容主要包括方法和靜態屬性。堆區域用來存放Class的實例(即對象),對象須要存儲的內容主要是非靜態屬性。每次用new建立一個對象實例後,對象實例存儲在堆區域中,這部分空間也被jvm的垃圾回收機制管理。而Java棧跟大多數編程語言包括彙編語言的棧功能類似,主要基本類型變量以及方法的輸入輸出參數。Java程序的每一個線程中都有一個獨立的堆棧。容易發生內存溢出問題的內存空間包括:Permanent Generation space和Heap space。web
發生這種問題的原意是程序中使用了大量的jar或class,使java虛擬機裝載類的空間不夠,與Permanent Generation space有關。解決這類問題有如下兩種辦法:算法
JAVA_OPTS=" -XX:PermSize=64M -XX:MaxPermSize=128m"
若是是windows服務器還能夠在系統環境變量中設置。感受用tomcat發佈sprint+struts+hibernate架構的程序時很容易發生這種內存溢出錯誤。使用上述方法,我成功解決了部署ssh項目的tomcat服務器常常宕機的問題。發生這種問題的緣由是java虛擬機建立的對象太多,在進行垃圾回收之間,虛擬機分配的到堆內存空間已經用滿了,與Heap space有關。解決這類問題有兩種思路:編程
set JAVA_OPTS= -Xms256m -Xmx1024m
在java應用中,有時候會出現這樣的錯誤:OutOfMemoryError: unable to create new native thread.這種怪事是由於JVM已經被系統分配了大量的內存(好比1.5G),而且它至少要佔用可用內存的一半。有人發現,在線程個數不少的狀況下,你分配給JVM的內存越多,那麼,上述錯誤發生的可能性就越大。windows
那麼是什麼緣由形成這種問題呢?tomcat
每個32位的進程最多可使用2G的可用內存,由於另外2G被操做系統保留。這裏假設使用1.5G給JVM,那麼還餘下500M可用內存。這500M內存中的一部分必須用於系統dll的加載,那麼真正剩下的也許只有400M,如今關鍵的地方出現了:當你使用Java建立一個線程,在JVM的內存裏也會建立一個Thread對象,可是同時也會在操做系統裏建立一個真正的物理線程(參考JVM規範),操做系統會在餘下的400兆內存裏建立這個物理線程,而不是在JVM的1500M的內存堆裏建立。在jdk1.4裏頭,默認的棧大小是256KB,可是在jdk1.5裏頭,默認的棧大小爲1M每線程,所以,在餘下400M的可用內存裏邊咱們最多也只能建立400個可用線程。服務器
這樣結論就出來了,要想建立更多的線程,你必須減小分配給JVM的最大內存。還有一種作法是讓JVM宿主在你的JNI代碼裏邊。架構
給出一個有關可以建立線程的最大個數的估算公式:ssh
(MaxProcessMemory - JVMMemory - ReservedOsMemory) / (ThreadStackSize) = Number of threads
對於jdk1.5而言,假設操做系統保留120M內存:jvm
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
對於這個異常咱們首先須要判斷下,發生內存溢出時進程中到底都有什麼樣的線程,這些線程是不是應該存在的,是否能夠經過優化來下降線程數; 另一方面默認狀況下java爲每一個線程分配的棧內存大小是1M,一般狀況下,這1M的棧內存空間是足足夠用了,由於在一般在棧上存放的只是基礎類型的數據或者對象的引用,這些東西都不會佔據太大的內存, 咱們能夠經過調整jvm參數,下降爲每一個線程分配的棧內存大小來解決問題,例如在jvm參數中添加-Xss128k
將線程棧內存大小設置爲128k。