Java中的OutOfMemoryError,即内存溢出,而无论是哪种内存溢出,都可以通过增加对应JVM内存空间解决:即修改对应内存参数的大小
根据内存区域的不同,存在多种情况,下面介绍最常见的几种情况:
1、方法区内存溢出
① jdk1.7及以前:java.lang.OutOfMemoryError: PremGen space
我们知道,根据 JVM 规范,JVM 内存共分为虚拟机栈、堆、方法区、程序计数器、本地方法栈五个部分。这里的“PermGen space”其实指的就是方法区,即Permanent Generation space(永久保存区域)。不过方法区和“PermGen space”又有着本质的区别。前者是JVM的规范,而后者则是JVM规范的一种实现,并且只有HotSpot才有“PermGen space”。由于方法区主要存储类的相关信息,所以对于动态生成类的情况比较容易出现永久代的内存溢出。
所以解决这类问题的方法为:增加java虚拟机中的XX:PermSize和XX:MaxPermSize参数的大小,其中XX:PermSize是初始永久保存区域大小,XX:MaxPermSize是最大永久保存区域大小。针对tomcat,可以在tomcat配置的虚拟机选项中增加如下内容:
-server -XX:PermSize=1024M -XX:MaxPermSize=1024m
以IDEA为例,具体操作如图所示:
② jdk1.8以后
JDK 1.8中参数PermSize和MaxPermSize已经失效,所以对于方法区,Java8之后的变化:
- 移除了永久代(PermGen),替换为元空间(Metaspace);
- 永久代中的 class metadata 转移到了 native memory(本地内存,而不是虚拟机);
- 永久代中的 interned Strings 和 class static variables 转移到了 Java heap;
- 永久代参数 (PermSize MaxPermSize) -> 元空间参数(MetaspaceSize MaxMetaspaceSize)
元空间的本质和永久代类似,都是对JVM规范中方法区的实现。不过元空间与永久代之间的最大区别在于:元空间并不在虚拟机中,而是使用本地内存。因此,默认情况下,元空间的大小仅受本地内存限制,但可以通过以下参数指定元空间的大小 :
-XX:MetaspaceSize,初始空间大小,达到该值就会触发垃圾收集进行类型卸载,同时GC会对改值进行调整:如果释放了大量的空间,就适当降低该值;如果释放了很少的空间,那么在不超过MaxMetaspaceSize时,适当提高该值。
-XX:MaxMetaspaceSize,最大空间,默认是没有限制的。
2、Heap space(堆区域)
OutOfMemoryError: Java heap space
发生这种问题的原因是java虚拟机创建的对象太多,在进行垃圾回收之间,虚拟机分配的到堆内存空间已经用满了,与Heap space有关。解决这类问题有两种思路:
1. 检查程序,看是否有死循环或不必要地重复创建大量对象。找到原因后,修改程序和算法;
2. 增加Java虚拟机中Xms(初始堆大小)和Xmx(最大堆大小)参数的大小。
set JAVA_OPTS= -Xms256m -Xmx1024m
3、Java Stacks(Java栈)
栈通常会遇到两种内存异常:
StackOverflowError
OutOfMemoryError
通过修改参数-Xss修改,方法同上