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