`

调优jvm内存,并解决OutofMemoryError,StackOverflowError等异常问题

阅读更多

JVM 调优,首先应从内 存开始,尤其是在真正的的web服务部署的时候。因为真正的web服务会比开发的时候花费更多的内 存,用来处理多用户并发的情况。本人多次吃过这方面的亏,所以整理一下,希望能给别人以帮助。

这个年头变啦,内 存变得如大白菜,每个新装的机器都2G以上的内 存,甚至4G,也不是什么新闻。而软件‘吃’内 存的情况则变化不大(除了VIsta)。但 JAVA诞生的时候可不是这样——95年,想来当年97年,64M的内 存还要500元,所以JVM初始化对内 存的要不能太大,而且也要考虑老机器的情况,毕竟现在JRE基本跑在每个人的机器上。但是JVM初始占用还停留在几年前的情况下,确实没有跟上软件和硬件的发展。而像Tomcat, JBoss, Eclipse(尤其安上MyEclipse插件后),也考虑到每台机器的内 存情况,所以初始话定义都很低,经常会抛内 存溢出Bug。

好,言归正传。我们先从解决bug开始,当Java程序申请内 存,超出VM可分配内 纯的时候,VM首先可能会GC,如果GC完还是不够,或者申请的直接超够VM可能有的,就会抛出内 存溢出异常。从VM规范中我们可以得到,一下几种异常。

java.lang.StackOverflowError:(很少)
java.lang.OutOfMemoryError:heap space(比较常见)
java.lang.OutOfMemoryError: PermGen space (经常出现)

java.lang.OutOfMemoryError: GC overhead limit exceeded(某项操作使用大量内存时发生)

以下分别解释一下,从最常见的开始:

java.lang.OutOfMemoryError: PermGen space 这个异常比较常见,是说JVM里的Perm内 存区的异常溢出,由于JVM在默认的情况下,Perm默认为64M,而很多程序需要大量的Perm区内 存,尤其使用到像Spring等框架的时候,由于需要使用到动态生成类,而这些类不能被GC自动释放,所以导致OutOfMemoryError: PermGen space异常。解决方法很简单,增大JVM的 -XX:MaxPermSize 启动参数,就可以解决这个问题,如过使用的是默认变量通常是64M[5.0 and newer: 64 bit VMs are scaled 30% larger; 1.4 amd64: 96m; 1.3.1 -client: 32m.],改成128M就可以了,-XX:MaxPermSize=128m。如果已经是128m(Eclipse已经是128m了),就改成 256m。我一般在服务器上为安全起见,改成256m。

java.lang.OutOfMemoryError:heap space或 其它OutOfMemoryError,这个异常实际上跟上面的异常是一个异常,但解决方法不同,所以分开来写。上面那个异常是因为JVM的perm区内 存区分少了引起的(JVM的内 存区分为 young,old,perm三种)。而这个异常是因为JVM堆内 存或者说总体分少了。解决方法是更改 -Xms -Xmx 启动参数,通常是扩大1倍。xms是管理启动时最小内 存量的,xmx是管里JVM最大的内 存量的。
注:OutOfMemoryError可能有很多种原因,根据JVM Specification, 可能有一下几种情况,我先简单列出。stack:stack分区不能动态扩展,或不足以生成新的线程。Heap:需要更多的内 存,而不能获得。Method Area :如果不能满足分配需求。runtime constant pool(从Method Area分配内 存)不足以创建class or interface。native method stacks不能够动态扩展,或生成新的本地线程。


java.lang.OutOfMemoryError: GC overhead limit exceeded,这个是JDK6新添的错误类型。是发生在GC占用大量时间为释放很小空间的时候发生的,是一种保护机制。我在JSP导大Excel的时候碰到过。最终解决方案是,关闭该功能,使用—— -XX:-UseGCOverheadLimit

最后说说java.lang.StackOverflowError,老实说这个异常我也没碰见过,但JVM Specification就提一下,规范上说有一下几种境况可能抛出这个异常,一个是Stacks里的线程超过允许的时候,另一个是当native method要求更大的内 存,而超过native method允许的内 存的时候。根据SUN的文档,提高-XX:ThreadStackSize=512的值。

总的来说调优JVM的内 存,组要目的就是在使用内 存尽可能小的,使程序运行正常,不抛出内 纯溢出的bug。而且要调好最小内 存,最大内 存的比,避免GC时浪费太多时间,尤其是要尽量避免FULL GC。


补充:由于JDK1.4新增了nio,而nio的buffer分配内存比较特殊(读写流可以共享内存)。如果有大量数据交互,也可能导致java.lang.OutOfMemoryError。相应的JDK新增了一个特殊的参数:-XX:MaxDirectMemorySize 默认是64M,可以改大些如128M。


-XX:MaxDirectMemorySize=<size>
Specifies the maximum amount of memory in bytes that the Java™ NIO library can allocate for direct memory buffers. The default is 64 megabytes, which corresponds to
-XX:MaxDirectMemorySize=64m . The use of direct memory buffers can minimize the copying cost when doing I/O operations.

 

以下是本文的参考文档:

http://java.sun.com/javase/technologies/hotspot/gc/gc_tuning_6.html#par_gc.oom
http://java.sun.com/docs/books/jvms/second_edition/html/Overview.doc.html
http://java.sun.com/j2se/1.5.0/docs/tooldocs/windows/java.html
http://java.sun.com/javase/technologies/hotspot/vmoptions.jsp

分享到:
评论

相关推荐

    如何解决java.lang.StackOverflowError

    在Java编程中,`java.lang.StackOverflowError` 是一个常见的运行时异常,它通常发生在程序执行过程中,当Java虚拟机(JVM)的调用栈溢出时。调用栈是每个线程用来存储方法调用信息的数据结构,当递归调用过深或者...

    Java中内存异常StackOverflowError与OutOfMemoryError详解

    Java中内存异常StackOverflowError与OutOfMemoryError是两种常见的错误类型,它们都是 Java 虚拟机(JVM)在运行时可能遇到的问题。本文将详细介绍这两种错误的定义、产生原因以及解决方法。 StackOverflowError ...

    JVM内存空间分配笔记

    ### JVM内存空间分配详解 #### 一、JVM内存模型概览 ...综上所述,理解JVM内存分配机制对于Java开发者来说至关重要,这不仅有助于编写高效、稳定的代码,还能在遇到性能瓶颈时快速定位问题并进行优化。

    JVM调优实践 ⼀、JVM调优准备⼯作 实验报告 pdf

    栈的大小有限制,深度递归可能导致`StackOverflowError`,而过多的线程则可能导致`OutOfMemoryError`。 2. **堆**:堆是所有线程共享的内存区域,用于存储对象实例和数组。大型数据存储在堆中,由栈中的引用指向。...

    JVM 内存管理及调优.zip_JVM内存_java_memory

    在Java 8之前,方法区的一部分被称为永久代,由于内存限制和垃圾回收问题,Java 8将其替换为Metaspace,以减少OutofMemoryError的发生。 3. **虚拟机栈**:每个线程都有一个独立的虚拟机栈,用于存储局部变量表、...

    jvm调优-jvm.zip

    `-Xss`参数可设置每个线程的栈大小,过大可能导致内存浪费,过小则可能引发StackOverflowError。 7. **编译优化**:JVM有即时编译器(JIT),会将频繁执行的热点代码编译为原生机器码。`-XX:CompileThreshold`和`-...

    JVM内存配置优化

    3. **`java.lang.StackOverflowError`**:这表示栈内存溢出,通常是由无限递归或者过深的递归导致的。 #### 三、Tomcat中的JVM配置优化 对于使用Tomcat的应用,可以通过修改其配置文件来进行JVM的优化。以`...

    JVM实战-对象访问与内存溢出异常解析

    通过本实验,旨在深入理解JVM内存管理机制以及各种内存区域的特点,并通过具体的编程实践来触发并分析这些异常,进而提升对Java应用程序性能调优和故障排查的能力。 #### 实验目标 1. **理解内存区域与内存区域...

    jvm内存溢出

    ### JVM内存溢出详解 #### 一、基本概念与理解 **JVM内存溢出**是一种常见的运行时错误,指的是程序在执行过程中因为...通过以上详细的分析和策略,可以有效地预防和解决JVM内存溢出问题,提高程序的稳定性和性能。

    java中jvm内存分配相关资料总结整理

    理解JVM的内存分配和运行原理是Java开发人员必备的技能,这有助于编写出更高效、更稳定的代码,并能有效地定位和解决性能问题。通过深入学习和实践,我们可以更好地驾驭Java虚拟机,提升应用程序的性能表现。

    JVM调优攻略.pdf

    《JVM调优攻略》是一份详尽的文档,旨在帮助开发者理解并掌握Java虚拟机(JVM)的优化技巧。本指南不仅适用于初学者,对于有一定基础的开发人员来说也同样具有很高的参考价值。文档中提到的核心概念包括JVM的工作原理...

    JVM调优总结:调优方法.docx

    - **堆栈溢出异常**:`java.lang.StackOverflowError`,通常由无限递归或循环调用引起,可以通过增大线程栈大小(`-Xss`)解决,但根源仍需在代码中排查。 - **系统内存满异常**:当系统内存被占满时,也会抛出`java....

    JVM内存分配与垃圾回收详解

    3. 本地方法栈:此区域与虚拟机栈起到的作用类似,只不是过虚拟机栈是针对 Java 方法而这个区域是针对 native 方法,同时这个区域也会抛出 StackOverflowError 和 OutOfMemoryError 两种异常。 4. Java 堆:它是 ...

    JVM内存有关知识点

    理解JVM内存管理对于优化代码性能、防止内存泄漏以及解决OutOfMemoryError等问题至关重要。以下是一些关于JVM内存的重要知识点: 1. **JVM内存结构**: JVM内存主要分为五个区域:方法区(Method Area)、堆(Heap...

    JVM内存模型及分区 JVM内存模型及分区

    Java虚拟机(JVM)内存模型是Java编程中不可或缺的一部分,它主要分为以下几个分区: ...此外,了解这些内存区域的工作原理也能帮助开发者在遇到栈溢出或内存溢出等问题时,更快地定位和解决问题。

    JVM内存区域划分.docx

    异常情况包括栈深度大于已有深度引发的StackOverflowError和可扩展深度大于能够申请的内存引发的OutOfMemoryError。 本地方法栈 本地方法栈与Java栈的作用和原理非常相似。区别只不过是Java栈是为执行Java方法服务...

    关于JVM内存溢出的原因分析及解决方案探讨.docx

    关于 JVM 内存溢出的原因分析及解决方案探讨 本文主要讨论了 JVM 中的内存溢出问题,包括内存溢出的定义、内存泄漏和...本文对 JVM 中的内存溢出问题进行了详细的分析和讨论,并提供了一些解决方案,以供读者参考。

    第25讲谈谈JVM内存区域的划分,哪些区域可能发生OutOfMemoryError1

    了解JVM内存区域的划分及其工作原理对于Java开发者来说至关重要,这有助于理解和解决性能问题,如内存泄漏和栈溢出。在实际开发中,合理设置JVM参数(如-Xms, -Xmx等)可以有效防止内存错误,同时,通过阅读如《深入...

Global site tag (gtag.js) - Google Analytics