目录

Java堆溢出

虚拟机栈和本地方法栈溢出

方法区溢出

直接内存溢出


Java堆溢出

  • 原因:创建的对象的总容量超过了堆的最大容量。
  • 溢出类型:java.lang.OutOfMemoryError。
  • 解决方法:
  1. 排查是因为内存溢出还是内存泄漏导致的:通过性能监测工具如jconsole,获取堆内存快照,查看导致溢出的对象是不是必要的,不是的话就是泄漏,是的话就是溢出。
  2. 内存泄漏:查看导致溢出的对象的GC ROOTS引用链,根据引用链找到具体的内存泄漏的位置,进行修改。
  3. 内存溢出:检查JVM的堆参数设置(-Xmx:JVM最大内存与-Xms:启动初始内存),看是否还有向上调整的空间;再从代码上检查是否有些对象的生命周期过长,或者存储上设计不合理等问题。

JVM中各内存区域溢出原因及解决办法

虚拟机栈和本地方法栈溢出

  • 原因:线程请求的栈的深度大于虚拟机所允许的最大深度(如不会结束的递归方法调用),将抛出StackOverflowError;当创建过多线程超过JVM内存上限时,会抛出OutOfMemoryError。
  • 溢出类型:java.lang.StackOverflowError或java.lang.OutOfMemoryError。
  • 解决方法:对于StackOverflowError,控制台会打印溢出错误的堆栈信息,比较容易定位;对于OutOfMemoryError,可以减少栈空间容量(-Xss设置每个线程的栈空间大小,调小这个参数可以创建更多的线程),或者减少线程数量。

JVM中各内存区域溢出原因及解决办法JVM中各内存区域溢出原因及解决办法

方法区溢出

  • 原因:方法区主要存储常量、对象类型数据。jdk1.6及之前,字符串常量池内存溢出(调用intern())导致方法区内存溢出,抛出OutOfMemoryError;创建过多的对象类型,典型地是在运行期间通过反射不断创建代理对象,导致方法区溢出,抛出OutOfMemoryError。
  • 溢出类型:java.lang.OutOfMemoryError。
  • 解决方法:jdk1.7开始,字符串常量池移到了堆中,不太可能出现内存溢出;jdk1.8开始,方法区也被元空间(Metaspace)取代,转移到了本地内存中,容量只受系统内存大小的限制,也不大可能出现内存溢出了。不过可以通过-XX:MaxMetaspaceSize指定元空间最大值,-XX:MetaspaceSize指定元空间大小初始值。

JVM中各内存区域溢出原因及解决办法JVM中各内存区域溢出原因及解决办法

直接内存溢出

  • 原因:直接内存中分配的数据的容量超过了上限(默认和Java堆最大值-Xmx一致,可以通过-XX:MaxDirectMEmorySize指定),抛出OutOfMemoryError,这个容量是计算出来的,并不是真的向操作系统申请不到后抛出的。
  • 溢出类型:java.lang.OutOfMemoryError。
  • 解决方法:排查可以看Heap Dump文件,如果Dump文件很小,并且程序中又使用了直接内存,那很可能就是直接内存溢出导致的OutOfMemoryError。

相关文章: