在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() 字节码,执行此方法执行,打印出结果。
下面再补充几个图,进一步了解: