栈
栈是一种数据结构,常用来和队列相比较。
栈是先进后出,队列是先进先出。
栈内存主管程序的运行,生命周期和线程同步。线程结束,栈内存也就释放了,不存在垃圾回收的问题。
栈里面存放的有8中基本类型、对象的引用、实例的方法。
栈的示意图如下,每执行完一个方法就会弹出一次,知道所有都弹出,线程结束。
有时会出现栈溢出的情况,是因为栈空间被占满,对应的代码如下:
public class Test {
//栈溢出测试
public static void main(String[] args) {
new Test().a();
}
public void a(){
b();
}
public void b(){
a();
}
}
结果输出:StackOverflowError(栈溢出)
抽象示意图如下:
对象实例化的过程示意图如下,通对对栈里面的引用对应堆中的对象实例化实体
堆
一个JVM中只有一个堆内存,堆内存的大小是可以调节的。
类加载器读取了类文件后,一般会把什么东西放到堆中?类、方法、常量、变量一般会被存放在堆中,堆中保存了所有引用类型的真实对象。
堆内存中分为三个区域:
新生区(伊甸园区)
老年区
永久区(1.8以后没了)
如果垃圾回收没有在新生区中被回收就会进入幸存区,称为轻GC,如果多次没被回收则会进入老年区,当老年区接近满的时候则会进行深度清理,称为重GC
GC回收主要发生在新生区(伊甸园区)和老年区,当内存满值的时候,则会报错(OOM),即堆内存不够。
在JDK8以后永久存储区改名为元空间。
新生区
- 新生区是一个类生成、成长、消亡的地方。
- 新生区主要分为伊甸园区、幸存1区,幸存2区。
- 所有对象都是在伊甸园区new出来的。
老年区
- 当经过新生区还未被杀死的对象会进入老年区。 研究表明,99%的对象都是临时对象,在新生区被kill。
永久区
这个区域是常驻内存的,用来存放JDK自身携带的Class对象,以及interface元数据,存放的是java运行时的一些环境或类信息。这个区域不存在垃圾回收,关闭虚拟机就会释放这个区域的内存。
什么情况下永久区会崩呢? 一个类加载大量第三方jar包、Tomcat部署太多应用,大量动态生成反射类不断地被加载直到内存满了就会出现OOM。
- dk1.6之前:永久代,常量池存在于方法区中。
- jdk1.7:永久代,慢慢的退化了,常量池在堆中。(去永久代)
- jdk1.8:无永久代,慢慢的退化了,常量池在元空间中。
在jdk1.8下堆中的结构如图所示:
测试虚拟机试图使用的最大内存和jvm初始化总内存?
默认情况下分配的总内存是电脑内存的1/4,初始化内存为默认为总内存的1/64
测试:
public class Test {
public static void main(String[] args) {
//返回虚拟机试图使用的最大内存
long max = Runtime.getRuntime().maxMemory(); //字节 1024*1024
//返回jvm的初始化总内存
long total = Runtime.getRuntime().totalMemory();
//默认情况下分配的最大内存是电脑内存的4/1,初始化内存是总的1/64
System.out.println("最大内存为:" + max/(double)1024/1024 + "MB");
System.out.println("初始化总内存为:" + total/(double)1024/1024 + "MB");
}
}
结果:
这里可以手动设置使用内存大小,如图所示
输入 -Xms1024m -Xmx1024m -XX:+PrintGCDetails //初始内存和总内存都调为1024
计算可得,年轻代和老年代内存的和等于内存值,因此可以得出,元空间在逻 辑上存在,在物理上不存在。
补充:
- -Xms :设置初始化内存分配大小
-Xmx:设置最大分配内存
-XX:+PrintGCDetails:打印GC垃圾回收信息
堆内存调优
当出现OOM时,可采取以下方法调优
- 尝试扩大堆内存 -Xms1024m -Xmx1024m -XX:+PrintGCDetails 查看结果。
若仍旧出现问题,分析内存,看一下哪里出现问题。
模拟堆内存溢出(OOM)
我们首先设置内存,将内存调小为1M。
修改内存 -Xms1m -Xmx1m -XX:+PrintGCDetails
测试:
public class Test {
public static void main(String[] args) {
String str = "123";
while (true){
str += new Random().nextInt(999999999);
}
}
}
运行结果:
首先轻GC,年轻代内存占满,然后重GC,老年代内存占满,最后内存溢出,OOM报错。
在一个项目中突然出现了OOM错误,那么该如何排除,研究为什么出错?
- 能够看到代码第几行出错:内存快照分析工具,MAT,Jprofiler
- Debug,一行行分析,先尝试扩大堆内存
-Xms1024m -Xmx1024m -XX:+PrintGCDetails
查看结果。若仍旧出现问题,分析内存,看一下哪里出现问题。