谈到堆、栈、方法区问题时,我们就会想到JVM内存模型。在本篇文章中,我也是结合几篇不错的博客进行的学习,并没有进行系统性的学习JVM。如有不当之处还望指出❤
1.JVM内存模型
Java虚拟机(Java Virtual Machine)内存空间分为五个部分,分别是:(加粗的为本章主要内容,其余两个作为了解哦)
堆区(heap):
- 堆是用来存放对象的内存空间,堆中不存放基本类型和对象引用,只存放对象本身
- jvm中只有一个堆区,被所有线程共享
- 在虚拟机启动时创建,垃圾回收的主要场所
- 堆的大小可以是固定的 ,也可以扩展。因此当线程请求分配内存,但堆已满且内存已满无法扩展时,就抛出OutOfMemoryError
栈区(stack):
- 基本类型变量区、执行环境上下文、操作指令区
- 每一个线程包含一个stack,随着线程的创建而创建,随着线程消亡而消亡。只保存基本数据类型和自定义对象的引用(不是对象)
- 每个stack中的数据都是私有的,其他栈不能使用
- Java虚拟机栈会出现两种异常:StackOverFlowError和OutOfMemoryError
- StackOverFlowError表示当前线程申请的栈超过了事先约定好栈的最大深度,且内存空间还有很多
- OutOfMemoryError表示线程申请栈时发现栈已经满了,而且内存也满了
方法区(method area):
- 方法区中包含的都是程序中唯一的元素,如class、static
- 又叫静态区,被所有线程共享
- 和堆一样,允许固定大小,也允许可扩展的大小,还允许不实现垃圾回收
程序计数器(Program Counter Register):
我们都知道JVM是一行一行执行字节码,所以需要计数器来记录当前执行的行号。特点包括以下几个方面:
- 是一块较小的存储空间
- 线程私有的,每个线程都有一个程序计数器
- 生命周期随着线程的创建而创建,随着线程的消亡而消亡
- 是唯一一个不会出现OutOfMemoryError的内存区域
本地方法栈(Native method Stack):
本地方法栈和JVM虚拟机栈实现的功能类似,只不过本地方法区是本地方法运行的内存模型。本地方法在执行时,会创建一个栈帧来存放方法的各种信息。比如局部变量和返回值。方法执行前就创建栈帧入栈,执行完成后就出栈。
以下是完整的虚拟机模型,如下图所示:
2.示例:
public class MainMemory {
int id,size;
String name;
Data data;
void store(){
System.out.println("执行存储中,存储类型数据:"+data.type);
}
void release(){
System.out.println("释放内存中");
}
public MainMemory(){
}
public static void main(String[] args) {
MainMemory mainMemory=new MainMemory();
mainMemory.id=1;
mainMemory.size=1024;
mainMemory.name="test";
Data data=new Data();
data.type="字符串";
mainMemory.data=data;
mainMemory.store();
mainMemory.release();
}
}
class Data{
public String type;
}
(1)系统收到指令,启动一个java虚拟机进程
(2)进程首先在classpath中找到MainMemory.class文件,读取该文件的二进制数据 ,将该类数据存放到方法区中,整个过程就是类加载过程。
(3)java虚拟机定位到MainMemory类的main方法中,执行该方法,会分配一个栈帧。
(4)创建一个实例对象,也就是在堆区分配一个对象的内存,给对象属性赋值
(5)调用store(),分配栈帧
这里我感觉有一个不错的总结
1.java虚拟机内存模型中一共有两个栈,分别是java虚拟机栈和本地方法栈。两个栈的功能类似,都是方法运行过程的内存模型,并且两个栈的内部结构相同,都是线程私有的。
2.Java虚拟机的内存模型中一共有两个“堆”,一个是原本的堆,一个是方法区。方法区本质上是属于堆的一个逻辑部分。堆中存放对象,方法区中存放类信息、常量、静态变量、即时编译器编译的代码。
3.堆是Java虚拟机中最大的一块内存区域,也是垃圾收集器主要的工作区域。
4.程序计数器、Java虚拟机栈、本地方法栈是线程私有的,即每个线程都拥有各自的程序计数器、Java虚拟机栈、本地方法区。并且他们的生命周期和所属的线程一样。
而堆、方法区是线程共享的,在Java虚拟机中只有一个堆、一个方法栈。并在JVM启动的时候就创建,JVM停止才销毁。
有补充的小伙伴,欢迎留言哦,共同学习 共同进步。bye~~~❤