目的:
1.通过代码验证JVM描述的各个运行时区域存储的内容
2.在工作中遇到实际的内存溢出时,根据异常信息快速定位是哪个区域的内存溢出
内存溢出(OutOfMemoryError)存在的区域:java虚拟机栈,本地方法栈,堆,方法区
一、堆内存溢出
堆容量参数设置:-Xms,-Xmx
堆溢出场景:堆是用来存储对象实例的,不断创建对象,并且保证GC Roots到对象之间有可达路径来避免垃圾回收机制清除这些对象,那么在对象数量达到最大堆就会产生内存溢出。
/**
* 设置最小堆为20M,设置最大堆为20M,则堆大小只能为20M,-XX:+HeapDumpOnOutOfMemoryError为分析快照
* -Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError
* @author scsch
*
* 说明:
* 1.使用while不断创建新对象
* 2.数组添加新增对象,保证GC Roots到对象之间有可达路径,
* 避免垃圾回收机制清楚对象
*/
public class HeapOOM {
static class OOMObject {
}
public static void main(String[] args) {
List<OOMObject> list = new ArrayList<>();
while (true) {
list.add(new OOMObject());
}
}
}
运行结果:
java.lang.OutOfMemoryError: Java heap space
Dumping heap to java_pid17788.hprof ...
Heap dump file created [28003175 bytes in 0.183 secs]
处理java内存问题的思路:
1.内存泄漏
查看泄露对象的GC Roots的引用链,找到泄漏对象通过怎样的路径与GC Roots相关联并导致垃圾收集器无法回收他们,可以快速定位泄漏代码的位置。
2.若非内存泄漏
内存中的对象都活着,检查虚拟机的堆参数-Xms与-Xmx,与机器物理内存对比看是否可以调大,或检查代码,看是否存在某些对象的生命周期过长、持有状态时间过长,减少程序运行时期的内存消耗。
二、虚拟机栈和本地方法栈溢出
栈容量参数:-Xss
栈溢出场景:
1.线程请求的栈深度 > JVM允许的栈深度,抛StackOverFlowError
2.JVM动态扩展时无法申请到足够的内存空间,抛OutOfMemoryError
/**
* -Xss128k
* @author scsch
*
*/
public class JavaVMStackSOF {
private int stackLength = 1;
public void stackLeak() {
stackLength++;
stackLeak();
}
public static void main(String[] args) throws Throwable {
JavaVMStackSOF oom = new JavaVMStackSOF();
try {
oom.stackLeak();
} catch (Throwable e) {
System.out.println("stack length:" + oom.stackLength);
throw e;
}
}
}
运行结果:
stack length:997
Exception in thread "main" java.lang.StackOverflowError
at com.uosfst.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:12)
at com.uosfst.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:13)
并且在上调栈大小时,栈深度变大。
实验结果:
单线程下,无论是栈帧太大还是虚拟机容量太小,都跑出StackOverFlowError。
多线程下,内存溢出异常与栈空间是否足够大不存在联系。每个线程分配的栈空间越大,能创建的线程越少。
三、方法区和运行时常量池溢出
方法区容量参数:-XX
内存溢出场景:
不断在常量池新建对象
/**
* -XX:PerSize=10M -XX:MaxPermSize=10M
* @author scsch
*
* intern:
* 如果字符串常量池中已经包含一个等于此String对象的字符串,则返回此对象,
* 否则将此String对象包含的字符串添加到常量池中,并且返回此String的引用
*
*/
public class RuntimeConstantPoolOOM {
public static void main(String[] args) {
List<String> list = new ArrayList<>();
int i = 0;
while (true) {
list.add(String.valueOf(i++).intern());
}
}
}
运行结构抛异常,提示PerGen space,但是我电脑跑不出结果。