Java中的OOM(Out of Memory)指java.lang.OutOfMemoryError错误。了解JVM的基本原理后,很容易理解以下几种常见的OOM。
这是最常见的OOM原因。
堆中主要存放各种对象实例,还有常量池等结构。当JVM发现堆中没有足够的空间分配给新对象时,抛出该异常。具体来讲,在刚发现空间不足时,会先进行一次Full GC,如果GC后还是空间不足,再抛出异常。
引起空间不足的原因主要有:
方法区主要存储类的元信息,实现在元数据区。当JVM发现元数据区没有足够的空间分配给加载的类时,抛出该异常。
引起元数据区空间不足的原因主要有:
但是元数据区被实现在堆外,主要受到进程本身的内存限制,这种实现下很难溢出。
jdk7中,方法区被实现在永久代中,错误原因同上。
永久代非常小,而且不会被回收,很容易溢出,因此,jdk8彻底废除了永久代,将方法区实现在元数据区。
以Linux系统为例,JVM创建的线程与操作系统中的线程一一对应,受到以下限制:
当无法在操作系统中继续创建线程时,抛出上述异常。
解决办法从原因中找:
默认配置下,如果GC花费了98%的时间,回收的内存都不足2%的话,抛出该异常。