栈:
1 基础数据类型 byte short int long float double char boolean
2 方法的形式参数,方法调用完后从栈空间回收
3 引用对象的地址,引用完后,栈空间地址立即被回收,堆空间等待GC
a) 栈内的数据线程之间独立
b) 具体细分为:
b.1) 基本类型变量区
b.2) 执行环境上下文
b.3) 操作指令区
堆
1 this
2 new出来的对象
3 数组
a) jvm只有一个堆区,并被所有线程共享。
方法区域(又叫 静态区)
1 字符串常量
2 static
3 所有的class
a) 被所有线程共享, 其内存放程序中永远唯一的元素,eg: 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()
字节码,执行此方法执行,打印出结果。
看另一个案例图,再次了解下 各个区域存放数据的位置: