java中三种常见内存溢出错误的处理方法
轉(zhuǎn)載自http://blog.csdn.net/zmken497300/article/details/52496189
相信有一定java開發(fā)經(jīng)驗(yàn)的人或多或少都會(huì)遇到OutOfMemoryError的問題,這個(gè)問題曾困擾了我很長時(shí)間,隨著解決各類問題經(jīng)驗(yàn)的積累以及對問題根源的探索,終于有了一個(gè)比較深入的認(rèn)識。
在解決java內(nèi)存溢出問題之前,需要對jvm(java虛擬機(jī))的內(nèi)存管理有一定的認(rèn)識。jvm管理的內(nèi)存大致包括三種不同類型的內(nèi)存區(qū)域:Permanent Generation space(永久保存區(qū)域)、Heap space(堆區(qū)域)、Java Stacks(Java棧)。其中永久保存區(qū)域主要存放Class(類)和Meta的信息,Class第一次被Load的時(shí)候被放入PermGen space區(qū)域,Class需要存儲的內(nèi)容主要包括方法和靜態(tài)屬性。堆區(qū)域用來存放Class的實(shí)例(即對象),對象需要存儲的內(nèi)容主要是非靜態(tài)屬性。每次用new創(chuàng)建一個(gè)對象實(shí)例后,對象實(shí)例存儲在堆區(qū)域中,這部分空間也被jvm的垃圾回收機(jī)制管理。而Java棧跟大多數(shù)編程語言包括匯編語言的棧功能相似,主要基本類型變量以及方法的輸入輸出參數(shù)。Java程序的每個(gè)線程中都有一個(gè)獨(dú)立的堆棧。容易發(fā)生內(nèi)存溢出問題的內(nèi)存空間包括:Permanent Generation space和Heap space。
第一種OutOfMemoryError: PermGen space
發(fā)生這種問題的原意是程序中使用了大量的jar或class,使java虛擬機(jī)裝載類的空間不夠,與Permanent Generation space有關(guān)。解決這類問題有以下兩種辦法:
第二種OutOfMemoryError: ?Java heap space
發(fā)生這種問題的原因是java虛擬機(jī)創(chuàng)建的對象太多,在進(jìn)行垃圾回收之間,虛擬機(jī)分配的到堆內(nèi)存空間已經(jīng)用滿了,與Heap space有關(guān)。解決這類問題有兩種思路:
第三種OutOfMemoryError:unable to create new native thread
在java應(yīng)用中,有時(shí)候會(huì)出現(xiàn)這樣的錯(cuò)誤:OutOfMemoryError: unable to create new native thread.這種怪事是因?yàn)镴VM已經(jīng)被系統(tǒng)分配了大量的內(nèi)存(比如1.5G),并且它至少要占用可用內(nèi)存的一半。有人發(fā)現(xiàn),在線程個(gè)數(shù)很多的情況下,你分配給JVM的內(nèi)存越多,那么,上述錯(cuò)誤發(fā)生的可能性就越大。
那么是什么原因造成這種問題呢?
每一個(gè)32位的進(jìn)程最多可以使用2G的可用內(nèi)存,因?yàn)榱硗?G被操作系統(tǒng)保留。這里假設(shè)使用1.5G給JVM,那么還余下500M可用內(nèi)存。這500M內(nèi)存中的一部分必須用于系統(tǒng)dll的加載,那么真正剩下的也許只有400M,現(xiàn)在關(guān)鍵的地方出現(xiàn)了:當(dāng)你使用Java創(chuàng)建一個(gè)線程,在JVM的內(nèi)存里也會(huì)創(chuàng)建一個(gè)Thread對象,但是同時(shí)也會(huì)在操作系統(tǒng)里創(chuàng)建一個(gè)真正的物理線程(參考JVM規(guī)范),操作系統(tǒng)會(huì)在余下的400兆內(nèi)存里創(chuàng)建這個(gè)物理線程,而不是在JVM的1500M的內(nèi)存堆里創(chuàng)建。在jdk1.4里頭,默認(rèn)的棧大小是256KB,但是在jdk1.5里頭,默認(rèn)的棧大小為1M每線程,因此,在余下400M的可用內(nèi)存里邊我們最多也只能創(chuàng)建400個(gè)可用線程。
這樣結(jié)論就出來了,要想創(chuàng)建更多的線程,你必須減少分配給JVM的最大內(nèi)存。還有一種做法是讓JVM宿主在你的JNI代碼里邊。
給出一個(gè)有關(guān)能夠創(chuàng)建線程的最大個(gè)數(shù)的估算公式:
(MaxProcessMemory?-?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?對于這個(gè)異常我們首先需要判斷下,發(fā)生內(nèi)存溢出時(shí)進(jìn)程中到底都有什么樣的線程,這些線程是否是應(yīng)該存在的,是否可以通過優(yōu)化來降低線程數(shù); 另外一方面默認(rèn)情況下java為每個(gè)線程分配的棧內(nèi)存大小是1M,通常情況下,這1M的棧內(nèi)存空間是足足夠用了,因?yàn)樵谕ǔT跅I洗娣诺闹皇腔A(chǔ)類型的數(shù)據(jù)或者對象的引用,這些東西都不會(huì)占據(jù)太大的內(nèi)存, 我們可以通過調(diào)整jvm參數(shù),降低為每個(gè)線程分配的棧內(nèi)存大小來解決問題,例如在jvm參數(shù)中添加-Xss128k將線程棧內(nèi)存大小設(shè)置為128k。
總結(jié)
以上是生活随笔為你收集整理的java中三种常见内存溢出错误的处理方法的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: SQL中Group By的使用,以及一些
- 下一篇: java.lang.reflect.Co