当我们在工作中,遇到线上问题:内存溢出,如何解决呢?除了物理扩大内存以外,还可以从软件角度去定位问题之所在。
补充一下基础知识(这里以jdk8为例)
Java的内存模型,也可以称之为:运行时数据区(规范)
运行时数据区分为:堆、程序计数器、方法区、虚拟机栈、本地方法栈。
Java内存结构(实现)分为:堆区和非堆区。堆区分为Young区和Old区。非堆区即Metaspace,可分为CCS(压缩类空间)和CodeCache(native code存放的内存空间),当然这两块空间不一定存在,要看是否添加了相应的配置。Young区又分为Eden区和Surviver区,对象分配初期都在Eden区上,随着年龄的增加,进入Survivor区和Old区。Surviver区又分为S0和S1两块。S0和S1两块区域是等大的,不同时使用。对象在这两块区域上不断交换,每换一次岁数加一。Eden区和Survivor区的大小比例默认为8:2,因为很多对象的存活时间非常短,用朝生夕死来比喻刚好恰当,所以Eden区要大一些。
好了,我们通过代码来实机演示一下内存溢出的定位过程。
1、环境介绍:
springboot 2.1.3.RELEASE,添加Web依赖、IDEA开发工具、MemoryAnalyzerTool工具
2、实现思路:这里我们通过指定内存大小,然后不断的new对象到集合中,最终实现内存溢出的效果。
3、代码截图: