JAVA的JVM的内存可分为3个区:堆(heap)、栈(stack)和方法区(method)
栈区:
每个线程包含一个栈区,栈中只保存方法中(不包括对象的成员变量)的基础数据类型和自定义对象的引用(不是对象),对象都存放在堆区中
每个栈中的数据(原始类型和对象引用)都是私有的,其他栈不能访问。
栈分为3个部分:基本类型变量区、执行环境上下文、操作指令区(存放操作指令)。
虚拟机栈:表示Java方法执行的内存模型,每调用一个方法就会为每个方法生成一个栈帧(Stack Frame),用来存储局部变量表、操作数栈、动态链接、方法出口等信息。每个方法被调用和完成的过程,都对应一个栈帧从虚拟机栈上入栈和出栈的过程。虚拟机栈的生命周期和线程是相同的
本地方法栈:一个支持native方法调用的JVM实现
**程序计数器:**程序计数器用来记录当前正在执行的指令
堆区: 存储的全部是对象实例,每个对象都包含一个与之对应的class的信息(class信息存放在方法区)。jvm只有一个堆区(heap)被所有线程共享,堆中不存放基本类型和对象引用,只存放对象本身,几乎所有的对象实例和数组都在堆中分配。
方法区: 又叫静态区,跟堆一样,被所有的线程共享。它用于存储已经被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
用一个十分简单的代码段模拟一下Java堆栈的运行过程:
public class AppTest {
publicstaticvoidmain(String[] args) {
Person person = new Person("张三"); person.sayHello();
}
}
class Person {
String name; publicPerson(String name) {
this.name = name;
}
public void sayHello() {
System.out.println("hello " + name);
}
}
新建一个main线程的一个栈
jvm在方法区寻找是否存在Person类的信息
如果没找到类加载器加载Person类信息到方法区,然后在堆中创建Person对象,并持有方法区的引用
把person添加到main主线程栈中,并指向堆上的Person实例。
执行person的sayhello方法,找到堆上的Person实例。
根据堆上Person的方法区引用,找到方法区上sayhello的字节码,然后执行完成。
下面用一段十分简单的代码段描述一下栈的运行过程。
public class AppTest {
public static void main(String[] args) {
int a = 1; int b = 2; int c = add(a , b);
}
privatestaticintadd(int a, int b) {
return a + b;
}
}