浅谈JVM内存溢出原因和解决思路
作者:JV 发布时间:2023-11-23 12:24:15
目录
栈溢出(虚拟机栈和本地方法栈)
产生原因
解决思路
堆溢出
产生原因
解决思路
方法区和运行时常量池溢出
产生原因
解决思路
本机直接内存溢出
产生原因
解决思路
栈溢出(虚拟机栈和本地方法栈)
产生原因
在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());
}
}
}
结果:
解决思路
首先通过内存映像分析工具确认是内存泄漏还是内存溢出。
如果是内存泄漏,说明导致OOM的对象不是必要的。进一步通过工具查看GC Roots引用链。一般可以比较精确的定位。
如果是内存溢出,对象是必须存活的,那就检查虚拟机的堆参数-Xms、-Xmx设置,对比机器内存,看是否还有上调的空间。再从代码上检查对象生命周期、持有状态时间、存储结构是否有设计不合理等情况。
方法区和运行时常量池溢出
产生原因
一个类要被垃圾收集器回收,条件是比较苛刻的。在经常运行时生成大量动态类的应用场景里,就应该特别关注了。
解决思路
HotSpot在JDK8中已经完全使用元空间代替永久带。Hotspot提供了一些参数作为元空间的防御措施,主要包括:
XX:MaxMetaspacesize:设置元空间最大值,默认是-1,即不限制,或者说只受限于本地内存大小。
-XX:Metaspacesize :指定元空间的初始空间大小,以宇节为单位,达到该值就会触发垃圾收集进行类型卸载,同时收集器会对该值进行调整:如果释放了大量的空间,就适当降低该值;如果释放了很少的空间,那么在不超过-XX:MaxMetaspaceSize(如果设置了的话)的情况下,适当提高该值。
-XX:MinMetaspace Free Ratio:作用是在垃圾收集之后控制最小的元空间剩余容量的百分比,可减少因为元空间不足导致的垃圾收集的频率。类似的还有-xx:Max-MetaspaceFreeRatio,用于控制最大的元空间剩余容量的百分比。
本机直接内存溢出
产生原因
在直接或间接使用了ByteBuffer中的allocateDirect方法的时候,而不做clear的时候就会出现类似的问题。明显的特征是在Heap Dump文件中不会看到明显的异常情况。
解决思路
设置参数: -XX:MaxDirectMemorySize
来源:https://juejin.cn/post/7045929460167606280


猜你喜欢
- 前言在foreach中删除元素时,每一次删除都会导致集合的大小和元素索引值发生变化,从而导致在foreach中删除元素时会抛出异常。集合已修
- Java中的多线程是一种抢占式的机制,而不是分时机制。抢占式的机制是有多个线程处于可运行状态,但是只有一个线程在运行。 共同点: 1. 他们
- 在开发中,我们会遇到将不同组织架构合并成tree这种树状结构,那么如果做呢?实际上,我们也可以理解为如何将拥有父子关系的list转成树形结构
- 原文:http://it.deepinmind.com/java/2015/03/17/20-examples-of-date-and-ti
- 前言更新都写完了,但是要更新文件要怎么操作呢?连接服务器然后上传上去,修改下xml的版本号当然也是可以的,但是还是没有写个程序使用起来方便,
- 我们都知道Android应用软件基本上都会用到登录注册功能,那么对一个一个好的登录注册模块进行封装就势在必行了。这里给大家介绍一下我的第一个
- 上一篇文章讲了如何获取所有联系人,这篇文章就讲下怎么保存联系人数据到本机通讯录。这里我就假设你已经拿到了要保存的联系人数据。 因为
- 本文实例讲述了C#实现自定义Dictionary类。分享给大家供大家参考。具体如下:1.关于MyDictionary类本文中实现的MyDic
- 概述Spring Boot 监控核心是 spring-boot-starter-actuator 依赖,增加依赖后, Spring Boot
- JPA设置表名和实体名,表字段与实体字段的对应首先 你的jpaProperties配置项中要有<prop key="hibe
- dll的编写,首先是打开VS新建一个C++的控制台程序,下一步后选择dll以及空文档即可。然后就是添加一个类添加一个方法。方法排头固定格式
- 本文实例为大家分享了Android使用Retrofit上传文件的具体代码,供大家参考,具体内容如下一、封装RetrofitManagerpu
- 首先选择保存图片的路径:saveFileDialog1.Title = "保存"; &
- 先给大家展示下效果图,如果大家感觉不错,请参考实现代码。思路1.下角Button的父View加入一个FrameLayout,也就是图中全屏透
- APP生命周期wpf项目目录中有一个App.xaml.cs文件,该文件中App是一个partical类,与之对应的另一partical部分在
- 一、MVC架构1、MVC是什么MVC是模型Model、视图View和控制器Controller的简称,是一种架构规范降低了业务逻辑与视图之间
- 本文实例为大家分享了android自定义圆形倒计时显示控件的具体代码,供大家参考,具体内容如下先上效果图 - 倒计时结束代码块at
- 详细步骤首先在pom.xml文件中做一些修改:之前打war包需要修改打包方式,这次不需要了,因为默认就是 jar 包指定最终打成jar包的名
- 概述异步这个概念在不同语境下有不同的解释,比如在一个单核CPU里开启两个线程执行两个函数,通常认为这种调用是异步的,但对于CPU来说它是单核
- 要获得打印机的状态,应该定义一个联合.enum PrinterStatus { 其他状态= 1, 未知, 空闲