浅谈JVM内存溢出原因和解决思路

 更新时间:2021年12月27日 15:00:33   作者:JV  
本文主要介绍了浅谈JVM内存溢出原因和解决思路,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

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

产生原因

  • 在HotSpot中,只能由-Xss参数来设定。因为在HotSpot中不区分虚拟机栈和本地方法栈的。
  • 栈溢出时会出现两种异常:StackOverflowError异常和OutOfMemoryError异常。
    • StackOverflowError异常因为线程请求的栈深度大于虚拟机允许的最大深度。
    • OutOfMemoryError异常发生在虚拟机栈内存允许动态扩展的情况下,当扩展栈容量无法申请到足够的内存时。
  • 因为HotSpot是不支持扩展的,所在除非在线程创建时申请内存无法满足时,才会出现OutOfMemoryError,其余都是产生StackOverflowError异常。
  • 结论:给每个线程的栈分配内存不是越大越好。可以这么理解,比如总的内存是2G,如果一个线程就占了1.5G,那就。。。。

解决思路

出现 StackOverflowError异常时,会有明确错误堆栈可供分析,相对而言比较容易定位到问题所在。

如果使用Hotspot虚拟机默认参数,栈深度在大多数情况下(因为每个方法压人栈的帧大小并不是一样的,所以只能说大多数情况下)到达1000~2000 是完全没有问题,对于正常的方法调用(包括不能做尾递归优化的递归调用),这个深度应该完全够用了。但是,如果是建立过多线程导致的内存滥出,在不能减少线程数量或者更换 64 位虚拟机的情况下,就只能通过减少最大堆和减少栈容量来换取更多的线程。

堆溢出

产生原因

当不断的创建对象并避免垃圾回收时,总容量触及最大堆容量时,就会产生溢出。
运行代码:设置vm参数-Xms10m -Xmx10m

public class HeapTest {
    static class OOMObj{

    }
    /**
     * vm arg -Xms10m -Xmx10m -XX:+HeapDumpOnOutOfMemoryError
     */

    public static void main(String[] args) {
        List<OOMObj> oomObjList = new ArrayList<OOMObj>();
        while (true){
            oomObjList.add(new OOMObj());
        }
    }
}

结果:

解决思路

首先通过内存映像分析工具确认是内存泄漏还是内存溢出。

  1. 如果是内存泄漏,说明导致OOM的对象不是必要的。进一步通过工具查看GC Roots引用链。一般可以比较精确的定位。
  2. 如果是内存溢出,对象是必须存活的,那就检查虚拟机的堆参数-Xms、-Xmx设置,对比机器内存,看是否还有上调的空间。再从代码上检查对象生命周期、持有状态时间、存储结构是否有设计不合理等情况。

方法区和运行时常量池溢出

产生原因

一个类要被垃圾收集器回收,条件是比较苛刻的。在经常运行时生成大量动态类的应用场景里,就应该特别关注了。

解决思路

HotSpot在JDK8中已经完全使用元空间代替永久带。Hotspot提供了一些参数作为元空间的防御措施,主要包括:

  1. XX:MaxMetaspacesize:设置元空间最大值,默认是-1,即不限制,或者说只受限于本地内存大小。
  2. -XX:Metaspacesize :指定元空间的初始空间大小,以宇节为单位,达到该值就会触发垃圾收集进行类型卸载,同时收集器会对该值进行调整:如果释放了大量的空间,就适当降低该值;如果释放了很少的空间,那么在不超过-XX:MaxMetaspaceSize(如果设置了的话)的情况下,适当提高该值。
  3. -XX:MinMetaspace Free Ratio:作用是在垃圾收集之后控制最小的元空间剩余容量的百分比,可减少因为元空间不足导致的垃圾收集的频率。类似的还有-xx:Max-MetaspaceFreeRatio,用于控制最大的元空间剩余容量的百分比。

本机直接内存溢出

产生原因

在直接或间接使用了ByteBuffer中的allocateDirect方法的时候,而不做clear的时候就会出现类似的问题。明显的特征是在Heap Dump文件中不会看到明显的异常情况。

解决思路

设置参数: -XX:MaxDirectMemorySize

到此这篇关于浅谈JVM内存溢出原因和解决思路的文章就介绍到这了,更多相关JVM内存溢出内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Springcloud基于OpenFeign实现服务调用代码实例

    Springcloud基于OpenFeign实现服务调用代码实例

    这篇文章主要介绍了Springcloud基于OpenFeign实现服务调用代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-08-08
  • MyBatis加载映射文件和动态代理的实现

    MyBatis加载映射文件和动态代理的实现

    本文主要介绍了MyBatis加载映射文件和动态代理的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-05-05
  • Spring Security配置保姆级教程

    Spring Security配置保姆级教程

    Spring Security是一个功能强大且可高度自定义的身份验证和访问控制框架。它是保护基于Spring的应用程序的事实上的标准。Spring Security是一个专注于为Java应用程序提供身份验证和授权的框架
    2023-02-02
  • java代码实现空间切割

    java代码实现空间切割

    大家好,本篇文章主要讲的是java代码实现空间切割,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下
    2022-01-01
  • 深入浅析Java中普通代码块、构造代码块与静态代码块

    深入浅析Java中普通代码块、构造代码块与静态代码块

    这篇文章主要介绍了Java中普通代码块、构造代码块与静态代码块的相关资料,静态代码块>Main()>构造代码块 。非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2016-08-08
  • 出现java.util.ConcurrentModificationException 问题及解决办法

    出现java.util.ConcurrentModificationException 问题及解决办法

    这篇文章主要介绍了出现java.util.ConcurrentModificationException 问题及解决办法的相关资料,需要的朋友可以参考下
    2017-02-02
  • java实现计算器模板及源码

    java实现计算器模板及源码

    这篇文章主要为大家详细介绍了java实现计算器模板及源码,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-06-06
  • 怎样给Kafka新增分区

    怎样给Kafka新增分区

    这篇文章主要介绍了怎样给Kafka新增分区问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-12-12
  • Java生成的随机数靠谱吗?多少次会重复?

    Java生成的随机数靠谱吗?多少次会重复?

    今天给大家带来的是关于Java的相关知识,文章围绕着Java生成的随机数靠不靠谱展开,文中有非常详细的介绍,需要的朋友可以参考下
    2021-06-06
  • springboot拦截器无法注入redisTemplate的解决方法

    springboot拦截器无法注入redisTemplate的解决方法

    在工作中我们经常需要做登录拦截验证或者其他拦截认证功能,但是在写拦截器的时候发现redisTemplate一直无法注入进来,本文就详细的介绍了解决方法,感兴趣的可以了解一下
    2021-06-06

最新评论