Java 内存溢出(java.lang.OutOfMemoryError)的常见情况和处理方式总结

来源:互联网 发布:网络语狗子是啥意思 编辑:程序博客网 时间:2024/06/14 22:20

在运行程序时候遇见java内存溢出,百度到一篇博客讲内存溢出方面非常详细,就先记录了下来,以后遇见其他问题可以看。

一、导致OutOfMemoryError异常的常见原因:

  1. 内存中加载的数据量过于庞大,如一次从数据库取出过多数据;
  2. 集合类中有对对象的引用,使用完后未清空,使得JVM不能回收;
  3. 代码中存在死循环或循环产生过多重复的对象实体;
  4. 使用的第三方软件中的BUG;
  5. 启动参数内存值设定的过小;

二、常见的错误提示:

  1. java.lang.OutOfMemoryError:Javaheapspace
  2. java.lang.OutOfMemoryError:GCoverheadlimitexceeded
  3. java.lang.OutOfMemoryError:PermGenspace
  4. java.lang.OutOfMemoryError:Directbuffermemory
  5. java.lang.OutOfMemoryError:unabletocreatenewnativethread
  6. java.lang.StackOverflowError

三、 Java常见的几种内存溢出及解决方法

【情况一】java.lang.OutOfMemoryError:Javaheapspace:

【解释】:这种是java堆内存不够,一个原因是真不够(如递归的层数太多等),另一个原因是程序中有死循环;
【解决方案】:如果是java堆内存不够的话,可以通过调整JVM下面的配置来解决:

 -Xms3062m   -Xmx3062m

【情况二】java.lang.OutOfMemoryError:GCoverheadlimitexceeded:

【解释】:JDK6新增错误类型,当GC为释放很小空间占用大量时间时抛出;一般是因为堆太小,导致异常的原因,没有足够的内存。
【解决方案】:
  1、查看系统是否有使用大内存的代码或死循环;
  2、通过添加JVM配置,来限制使用内存:

-XX:-UseGCOverheadLimit

【情况三】java.lang.OutOfMemoryError:PermGenspace:

【解释】:这种是P区内存不够。
【解决方案】:可通过调整JVM的配置:

 -XX:MaxPermSize=128m   -XXermSize=128m

  【注】:
  JVM的Perm区主要用于存放Class和Meta信息的,Class在被Loader时就会被放到PermGenspace,这个区域成为年老代,GC在主程序运行期间不会对年老区进行清理,默认是64M大小,当程序需要加载的对象比较多时,超过64M就会报这部分内存溢出了,需要加大内存分配,一般128m足够。

【情况四】:java.lang.OutOfMemoryError:Directbuffermemory

【解决方案】:调整-XX:MaxDirectMemorySize=参数,如添加JVM配置:

 -XX:MaxDirectMemorySize=128m

【情况五】:java.lang.OutOfMemoryError:unabletocreatenewnativethread

【解释】:Stack空间不足以创建额外的线程,要么是创建的线程过多,要么是Stack空间确实小了。
【解决方案】:由于JVM没有提供参数设置总的stack空间大小,但可以设置单个线程栈的大小;而系统的用户空间一共是3G,除了Text/Data/BSS/MemoryMapping几个段之外,Heap和Stack空间的总量有限,是此消彼长的。因此遇到这个错误,可以通过两个途径解决:
1.通过-Xss启动参数减少单个线程栈大小,这样便能开更多线程(当然不能太小,太小会出现StackOverflowError);
2.通过-Xms-Xmx两参数减少Heap大小,将内存让给Stack(前提是保证Heap空间够用)。

【情况六】:java.lang.StackOverflowError

【解释】:这也内存溢出错误的一种,即线程栈的溢出,要么是方法调用层次过多(比如存在无限递归调用),要么是线程栈太小。
【解决方案】:优化程序设计,减少方法调用层次;调整-Xss参数增加线程栈大小。

四、java可设置最大内存

测试方法:在命令行下用 java -XmxXXXXM -version 命令来进行测试,然后逐渐的增大XXXX的值,如果执行正常就表示指定的内存大小可用,否则会打印错误信息。

(1)堆(Heap)和非堆(Non-heap)内存

按照官方的说法:“Java 虚拟机具有一个堆,堆是运行时数据区域,所有类实例和数组的内存均从此处分配。堆是在 Java 虚拟机启动时创建的。”“在JVM中堆之外的内存称为非堆内存(Non-heap memory)”。可以看出JVM主要管理两种类型的内存:堆和非堆。简单来说堆就是Java代码可及的内存,是留给开发人员使用的;非堆就是JVM留给自己用的,所以方法区、JVM内部处理或优化所需的内存(如JIT编译后的代码缓存)、每个类结构(如运行时常数池、字段和方法数据)以及方法和构造方法的代码都在非堆内存中。

(2)堆内存分配

JVM初始分配的内存由-Xms指定,默认是物理内存的1/64;JVM最大分配的内存由-Xmx指定,默认是物理内存的1/4。默认空余堆内存小于40%时,JVM就会增大堆直到-Xmx的最大限制;空余堆内存大于70%时,JVM会减少堆直到-Xms的最小限制。因此服务器一般设置-Xms、-Xmx相等以避免在每次GC 后调整堆的大小。

(3)非堆内存分配

JVM使用-XX:PermSize设置非堆内存初始值,默认是物理内存的1/64;由XX:MaxPermSize设置最大非堆内存的大小,默认是物理内存的1/4

(4)JVM内存限制(最大值)

首先JVM内存限制于实际的最大物理内存(废话!呵呵),假设物理内存无限大的话,JVM内存的最大值跟操作系统有很大的关系。简单的说就32位处理器虽然可控内存空间有4GB,但是具体的操作系统会给一个限制,这个限制一般是2GB-3GB(一般来说Windows系统下为1.5G-2G,Linux系统下为2G-3G),而64bit以上的处理器就不会有限制了。

所以说设置VM参数导致程序无法启动主要有以下几种原因:

1) 参数中-Xms的值大于-Xmx,或者-XX:PermSize的值大于-XX:MaxPermSize;
2) -Xmx的值和-XX:MaxPermSize的总和超过了JVM内存的最大限制,比如当前操作系统最大内存限制,或者实际的物理内存等等。说到实际物理内存这里需要说明一点的是,如果你的内存是1024MB,但实际系统中用到的并不可能是1024MB,因为有一部分被硬件占用了。

阅读全文
0 0
原创粉丝点击