内存溢出(Out Of Memory)经常简称为OOM,在jvm中主要分为方法区、堆、栈、本地方法栈、程序计数器这几部分,其中程序计数器是唯一不会出现OOM的,也就是说其他区域都会出现OOM。下面来分别说明几个区域出现OOM的情况及特征。

1. java堆溢出(heap)

Java堆内存主要用来存放运行过程中所有的对象,该区域OOM异常一般会有如下错误信息:

java.lang.OutofMemoryError:Java heap space

可以通过dump的内存快照就能分析,到底是由于程序原因导致的内存泄露,还是由于没有估计好JVM内存的大小而导致的内存溢出。

2. 栈溢出(stack)

栈用来存储线程的局部变量表、操作数栈、动态链接、方法出口等信息。如果请求栈的深度不足时抛出的错误会包含类似下面的信息: java.lang.StackOverflowError。

由于每个线程占的内存大概为1M,因此线程的创建也需要内存空间。如果申请创建的线程比较多超过剩余内存的时候,也会抛出如下类似错误:java.lang.OutofMemoryError: unable to create new native thread

与栈相关的JVM参数有:

1、 -Xss: 每个线程的堆栈大小,JDK5.0以后每个线程堆栈大小为1M,以前每个线程堆栈大小为256K.

2、 在相同物理内存下,减小这个值能生成更多的线程.但是操作系统对1个进程内的线程数还是有限制的,不能无限生成。

出现栈溢出可能原因:

递归:递归里用到的局部变量存储在堆栈中,堆栈的访问效率高,速度快,但空间有限,递归太多变量需要一直入栈而不出栈,导致需要的内存空间大于堆栈的空间,栈空间是2M,堆空间内存空间。

运行时常量溢出(constant)

运行时常量保存在方法区,存放的主要是编译器生成的各种字面量和符号引用,但是运行期间也可能将新的常量放入池中,比如String类的intern方法。如果该区域OOM,错误结果会包含类似下面的信息:

1 java.lang.OutofMemoryError: PermGen space

相关的JVM参数有:

1、 -XX:PermSize:设置持久代(perm gen)初始值,默认值为物理内存的1/64

2、 -XX:MaxPermSize:设置持久代最大值,默认为物理内存的1/4

方法区溢出

方法区主要存储被虚拟机加载的类信息,如类名、访问修饰符、常量池、字段描述、方法描述等。理论上在JVM启动后该区域大小应该比较稳定,但是目前很多框架,比如Spring和Hibernate等在运行过程中都会动态生成类,因此也存在OOM的风险。如果该区域OOM,错误结果会包含类似下面的信息:

java.lang.OutofMemoryError: PermGen space

相关的JVM参数有:

1 -XX:PermSize:设置持久代(perm gen)初始值,默认值为物理内存的1/64

2 -XX:MaxPermSize:设置持久代最大值,默认为物理内存的1/4