前言
基本所有程序员在开发工作中都会遇到OOM问题,都会面对堆区,栈区各种花式内存泄漏的情况。所以熟悉JVM内存模型结构是必须的了,以下我会带你慢慢了解清楚。
一、整体结构
1.结构简图介绍
啥都别说,直接看图~
JVM内存结构分为两部分:线程共享区和线程独占区
线程共享区
- 堆区和元区间(方法区)
线程独占区
- 栈区和本地方法栈以及程序计数器(pc寄存器)
2. 工作流程
简述一下jvm基本流程:
-
启动jvm后,通过类加载器加载编译好的class文件,并在将数据信息放在方法区中,同时在堆中创建一个这个类的Java.lang.Class对象。
-
当类在加载时,把会静态变量放到方法区,并赋初始值。
-
当实例化对象时,会在堆区分配空间,创建和存放对象信息。
-
当调用方法时,会使用到栈区
计数器用来
二、组成介绍
1.堆
Java堆区是java内存中所占最大的一块区域,几乎所有的java对象的实例都会被分配内存。对应的CG回收器也是主要管理这个模块的内存回收。
堆区分为:
- 年轻代(Young Generation)、老年代(Old Generation)。
- 默认情况下按照2:1比例。
年轻代又可以在细分为:
- Eden空间、From Survivor空间、To Survivor空间。
- 默认情况下年轻代按照8:1:1的比例来分配。
简述:
- 新创建的对象会分配在年轻代(Young Generation)中,在经历一次GC回收后,没有被回收的对象,会转移到From/To 区,并且对应的年龄+1。当经过多次gc后,年龄达到15的对象,会被转移到年老代(Old Generation)。
如果在堆中没有内存完成实例分配,并且堆也无法再扩展时,将会抛出OutOfMemoryError异常。
2.栈
虚拟机栈
- Java虚拟机栈(Java Virtual Machine Stacks)也是线程私有的,它的生命周期与线程相同。
- 每个方法被执行的时候都会同时创建一个栈帧(Stack Frame)用于存储局部变量表、操作栈、动态链接、方法出口等信息。
- 每一个方法的生命周期,就对应着一个栈帧在虚拟机栈中从入栈到出栈的过程。
本地方法栈
- 虚拟机栈是服务于java的方法调度。
- 而本地方法栈提供Native方法服务,可以调度其他语言的方法实现。
看下面的图,栈的结构组成~
- 局部变量表:存放局部变量的地址。
- 操作数栈:进行参数传递
- 动态链接:指向下一个栈帧或者方法区静态变量
- 返回地址:执行完后,返回上一个栈帧
异常:
如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常;如果虚拟机栈可以动态扩展(当前大部分的Java虚拟机都可动态扩展,只不过Java虚拟机规范中也允许固定长度的虚拟机栈),当扩展时无法申请到足够的内存时会抛出OutOfMemoryError异常。
3. 元区间(方法区)
- 各个线程共享的内存区域
- 它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
4. 程序计数器
- 计数器记录的是正在执行的虚拟机字节码指令的地址;
- 虚拟机通过程序计数器,来确定下一条要执行的字节码指令。
总结
主要内容:
- jvm内存模型的组成
- 理解各个组成部分的作用