什么是OOM?
说白了就是内存用完了
JVM没有足够的内存来为对象分配空间并且垃圾回收器也已经没有空间可回收时,就会抛出这个error(说明应用程序已经无法处理这种问题)
那么,为什么会没有内存了呢???
-
分配的少了:比如虚拟机本身可使用的内存(一般通过启动时的VM参数指定)太少。
-
应用占用的太多,并且用完没释放,浪费了。此时就会造成内存泄漏或者内存溢出。
OOM会造成以下情况:
内存泄露:
- 申请使用完的内存没有释放,导致虚拟机不能再次使用该内存,此时这段内存就泄露了,因为申请者不用了,而又不能被虚拟机分配给别人用。
内存溢出:
- 申请的内存超出了JVM能提供的内存大小,此时称之为溢出。
常见OOM有:
-
java.lang.OutOfMemoryError: Java heap space
- java堆内存溢出,此种情况最常见,一般由于内存泄露或者堆的大小设置不当引起。
- 对于内存泄露,需要通过内存监控软件(JPofiler)查找程序中的泄露代码,而堆大小可以通过虚拟机参数-Xms,-Xmx等修改。
-
java.lang.OutOfMemoryError: PermGen space
- java永久代溢出,即方法区溢出了,一般出现于大量Class或者jsp页面,或者采用cglib等反射机制的情况,因为上述情况会产生大量的Class信息存储于方法区。此种情况可以通过更改方法区的大小来解决,使用类似-XX:PermSize=64m -XX:MaxPermSize=256m的形式修改。另外,过多的常量尤其是字符串也会导致方法区溢出。
-
java.lang.StackOverflowError
- 不会抛OOM error,但也是比较常见的Java内存溢出。JAVA虚拟机栈溢出,一般是由于程序中存在死循环或者深度递归调用造成的,栈大小设置太小也会出现此种溢出。可以通过虚拟机参数-Xss来设置栈的大小。
OOM分析—heapdump
当出现oom时,不要慌,下面来说解决方法
首先我的开发工具是IDEA,所以选择的工具是JPofiler工具
如果你的开发工具是Eclipse,那么你可以选择MAT工具
例如 堆OOM(java.lang.OutOfMemoryError: Java heap space)
- 在IDEA设置JVM参数-XX:+HeapDumpOnOutOfMemoryError,设定当发生OOM时自动dump出堆信息
- 当发生堆oom时,会在项目的目录下生成相对应的dump文件
- 将文件在JPofiler工具中打开
- 定位异常行的代码