jvm堆溢出和栈溢出
一、jvm堆溢出
1、介绍
在jvm运行java程序时,如果程序运行所需要的内存大于系统的堆最大内存(-Xmx),就会出现堆溢出问题。
2、案例
// 介绍:执行该段代码需要大于10m内存空间
public class HeadOverflow {
public static void main(String[] args) {
List<Object> listObj = new ArrayList<Object>();
for(int i=0; i<10; i++){
Byte[] bytes = new Byte[1*1024*1024];
listObj.add(bytes);
}
System.out.println("添加success");
}
}
// 设置该程序的jvm参数信息
-Xms1m -Xmx10m -XX:+PrintGCDetails -XX:+HeapDumpOnOutOfMemoryError
初始堆内存和最大可以堆内存 Gc详细日志信息
3、总结
在正式项目部署环境程序默认读取的是系统的内存,一般设置程序的堆初始内存(-Xms) == 堆最大可用内存(-Xmx)。
二、jvm栈溢出
1、介绍
a、线程请求的栈深度大于虚拟机允许的最大深度 StackOverflowError
b、虚拟机在扩展栈深度时,无法申请到足够的内存空间 OutOfMemoryError
2、案例
// 循环递归调用,一直达到jvm的最大深度
public class StackOverflow {
private static int count;
public static void count(){
try {
count++;
count();
} catch (Throwable e) {
System.out.println("最大深度:"+count);
e.printStackTrace();
}
}
public static void main(String[] args) {
count();
}
}
设置-Xss5m设置最大调用深度后调用
总结:每个计算机都会有一个极限最大调用深度,避免递归在代码中无限循环。
三、内存溢出和内存泄露
1、区别
内存溢出:申请内存空间,超出最大堆内存空间。
内存泄露:其实包含内存溢出,堆内存空间被无用对象占用没有及时释放,导致占用内存,最终导致内存泄露。
情况:静态static修饰对象。
解决:减少常量的定义(具体看服务器内存情况)