java中三种常见内存溢出错误的处理方法
轉載自http://blog.csdn.net/zmken497300/article/details/52496189
相信有一定java開發經驗的人或多或少都會遇到OutOfMemoryError的問題,這個問題曾困擾了我很長時間,隨著解決各類問題經驗的積累以及對問題根源的探索,終于有了一個比較深入的認識。
在解決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。
第一種OutOfMemoryError: PermGen space
發生這種問題的原意是程序中使用了大量的jar或class,使java虛擬機裝載類的空間不夠,與Permanent Generation space有關。解決這類問題有以下兩種辦法:
第二種OutOfMemoryError: ?Java heap space
發生這種問題的原因是java虛擬機創建的對象太多,在進行垃圾回收之間,虛擬機分配的到堆內存空間已經用滿了,與Heap space有關。解決這類問題有兩種思路:
第三種OutOfMemoryError:unable to create new native thread
在java應用中,有時候會出現這樣的錯誤:OutOfMemoryError: unable to create new native thread.這種怪事是因為JVM已經被系統分配了大量的內存(比如1.5G),并且它至少要占用可用內存的一半。有人發現,在線程個數很多的情況下,你分配給JVM的內存越多,那么,上述錯誤發生的可能性就越大。
那么是什么原因造成這種問題呢?
每一個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代碼里邊。
給出一個有關能夠創建線程的最大個數的估算公式:
(MaxProcessMemory?-?JVMMemory?-?ReservedOsMemory)?/?(ThreadStackSize)?=?Number?of threads對于jdk1.5而言,假設操作系統保留120M內存:
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。
總結
以上是生活随笔為你收集整理的java中三种常见内存溢出错误的处理方法的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: SQL中Group By的使用,以及一些
- 下一篇: java.lang.reflect.Co