在Java程序运行过程中,JVM定义了各种区域用于存储运行时数据。其中的有些数据区域在JVM启动时创建,并只在JVM退出时销毁。其它的数据区域与每个线程相关。这些数据区域,在线程创建时创建,在线程退出时销毁。
JVM 由3部分组成:
栈区
---------------------------------
基础数据类型 byte short int long float double char boolean
方法的形式参数,方法调用完后从栈空间回收
引用对象的地址,引用完后,栈空间地址立即被回收,堆空间等待GC
栈内的数据线程之间独立
具体划分为:基本类型变量区、执行环境上下文、操作指令区
堆区
---------------------------------
this
new 出来的对象
数组
jvm只有一个堆区,并被所有线程共享
方法区(又叫静态区和永久区)
---------------------------------
字符串常量
static
所有class
被所有线程共享, 其内存放程序中永远唯一的元素
案例分析:
public class AppMain {// 运行时 jvm把类信息加入方法区
/**
* @param args
*/
public static void main(String[] args) {// main方法放入方法区
Sample test1 = new Sample("hello"); // test1是引用 放入栈中 new Sample("hello")对象放入堆中
test1.printName();
}
}
class Sample { // 运行时 jvm把 类信息放入方法区
private String name; // new Sample()后, name 引用放入栈 name对象放入堆
public Sample(String name){
this.name = name;
}
public void printName(){ // 方法本身放入方法区
System.out.println(name);
}
}
执行分析说明:
Sample test1 = new Sample("hello");
test1.printName();
0 知识前导:
jvm每个线程都拥有一个方法调用栈,用于跟踪线程中运行的一系列方法调用过程,
栈中的每个元素成为栈帧,线程调用每个方法时会将方法栈压入一个新帧,
帧里面存放方法参数,局部变量,运算过程产生的临时数据
1 jvm去方法区寻Sample类信息
2 寻找不到,jvm使用classloader加载Sample类信息进入内存方法区
3 在堆内存中创建Sample对象,并持有方法区中Sample类的类型信息的引用
4 test1添加到执行main()方法的主线程java调用栈中,指向堆空间中的内存对象
5 执行test1.printName()时,jvm根据test1定位到堆空间的Sample实例,在根据
Sample实例在方法区持有的引用,定位到方法区Sample类型信息,获得printName()
字节码,执行此方法执行,打印出结果。
下面再补充几个图,进一步了解: