JVM 内存 (堆(heap)、栈(stack)和方法区(method) )
JVM 内存初学 (堆(heap)、栈(stack)和方法区(method) )
堆区:
1.存储的全部是对象,每个对象都包含一个与之对应的class的信息。(class的目的是得到操作指令)
2.jvm只有一个堆区(heap)被所有线程共享,堆中不存放基本类型和对象引用,只存放对象本身
栈区:
1.每个线程包含一个栈区,栈中只保存基础数据类型的对象和自定义对象的引用(不是对象),对象都存放在堆区中。
2.每个栈中的数据(原始类型和对象引用)都是私有的,其他栈不能访问。
3.栈分为3个部分:基本类型变量区、执行环境上下文、操作指令区(存放操作指令)。
栈帧:一个栈帧随着一个方法的调用开始而创建,调用完成而销毁。每个方法从调用到结束就对于一个栈桢在虚拟机栈中的入栈和出栈过程
JVM栈只对栈帧进行存储,压栈和出栈操作。Java栈是Java方法执行的内存模型。
每个方法涉及的局部变量表和操作数栈的大小取决于每个具体的方法,但是大小在编译后便已确定,而且已经包含在class文件中。
- 每个JVM线程有一个私有栈,栈在线程创建的同时被创建。
- 栈由许多帧组成,也叫 “栈帧”
- 每次方法调用都会创建一个栈帧
方法区:
1.又叫静态区,跟堆一样,被所有的线程共享。方法区包含所有的class和static变量。
2.方法区中包含的都是在整个程序中永远唯一的元素,如class,static变量。
为了更清楚地搞明白发生在运行时数据区里的黑幕,我们来准备2个小道具(2个非常简单的小程序)。
1 AppMain.java 2 public class AppMain{//运行时, jvm 把appmain的信息都放入方法区 3 public static void main(String[] args){ //main 方法本身放入方法区。 4 Sample test1 = new Sample( " 测试1 " ); //test1是引用,所以放到栈区里,Sample是自定义对象应该放到堆里面 5 Sample test2 = new Sample( " 测试2 " ); 6 test1.printName(); 7 test2.printName(); 8 } 9 } 10 11 Sample.java 12 public class Sample { //运行时, jvm 把appmain的信息都放入方法区 13 /** 范例名称 */ 14 private String name; //new Sample实例后,name引用放入栈区里,name对象放入堆里 15 /** 构造方法 */ 16 public Sample(String name) 17 { 18 this .name = name; 19 } 20 /** 输出 */ 21 public void printName() //print方法本身放入 方法区里。 22 { 23 System.out.println(name); 24 } 25 }
类的加载过程
系统收到了我们发出的指令,启动了一个Java虚拟机进程,这个进程首先从classpath中找到AppMain.class文件,读取这个文件中的二进制数据,然后把Appmain类的类信息存放到运行时数据区的方法区中。这一过程称为AppMain类的加载过程。
接着,Java虚拟机定位到方法区中AppMain类的Main()方法的字节码,开始执行它的指令。这个main()方法的第一条语句就是:
Sample test1=new Sample(“测试1”);
让java虚拟机创建一个Sample实例,使引用变量test1引用这个实例。
Java虚拟机执行过程
跟踪一下Java虚拟机,看它是怎么来执行这个任务的:
1、建立一个Sample实例:到方法区,先找到Sample类的类型信息。结果这会儿的方法区里还没有Sample类呢。于是,Java虚拟机发扬“自己动手,丰衣足食”,立马加载了Sample类,把Sample类的类型信息存放在方法区里。
2、Java虚拟机在堆区中为一个新的Sample实例分配内存, 这个Sample实例持有着指向方法区的Sample类的类型信息的引用。这里所说的引用,实际上指的是Sample类的类型信息在方法区中的内存地址
【堆中Sample实例持有着指向方法区的Sample类的内存地址】,其实,就是有点类似于C语言里的指针啦~~,而这个地址呢,就存放了在Sample实例的数据区里。
3、 在JAVA虚拟机进程中,每个线程都会拥有一个方法调用栈,用来跟踪线程运行中一系列的方法调用过程,栈中的每一个元素就被称为栈帧,每当线程调用一个方法的时候就会向方法栈压入一个新帧。这里的帧用来存储方法的参数、局部变量和运算过程中的临时数据。
位于“=”前的Test1是一个在main()方法中定义的变量,可见,它是一个局部变量,因此,它被会添加到了执行main()方法的主线程的JAVA方法调用栈中。而“=”将把这个test1变量指向堆区中的Sample实例,也就是说,它持有指向Sample实例的引用。
接下来,JAVA虚拟机将继续执行后续指令,在堆区里继续创建另一个Sample实例,然后依次执行它们的printName()方法。当JAVA虚拟机执行test1.printName()方法时,JAVA虚拟机根据局部变量test1持有的引用,定位到堆区中的Sample实例,再根据Sample实例持有的引用,定位到方法区中Sample类的类型信息,从而获得printName()方法的字节码,接着执行printName()方法包含的指令。