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为例,具体操作如图所示: 

java 字节 缓冲区 java缓冲区溢出_jvm

② jdk1.8以后

JDK 1.8中参数PermSize和MaxPermSize已经失效,所以对于方法区,Java8之后的变化:

  1. 移除了永久代(PermGen),替换为元空间(Metaspace);
  2. 永久代中的 class metadata 转移到了 native memory(本地内存,而不是虚拟机);
  3. 永久代中的 interned Strings 和 class static variables 转移到了 Java heap;
  4. 永久代参数 (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修改,方法同上