JVM学习笔记(三)栈,堆

本文详细介绍了Java内存的栈、堆和元空间的概念,以及它们在程序运行中的作用。栈主要用于存放基本类型、对象引用和方法,而堆则是对象实例化的存储区域,分为新生区、老年区和永久区(在JDK1.8后变为元空间)。文章还讨论了栈溢出和堆内存溢出(OOM)的情况,并提供了示例代码进行说明。同时,提到了如何通过调整JVM参数进行内存调优,以及在出现OOM时的排查方法。
摘要由CSDN通过智能技术生成

栈是一种数据结构,常用来和队列相比较。
栈是先进后出,队列是先进先出。
栈内存主管程序的运行,生命周期和线程同步。线程结束,栈内存也就释放了,不存在垃圾回收的问题。
栈里面存放的有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 查看结果。若仍旧出现问题,分析内存,看一下哪里出现问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值