Java虚拟机的内存可以分为三个区域:栈stack、堆heap、方法区method area。
首先先讲讲上面三个的特点;
栈的特点如下:
1. 栈描述的是方法执行的内存模型。每个方法被调用都会创建一个栈帧(存储局部变量、操作数、方法出口等)
2. JVM为每个线程创建一个栈,用于存放该线程执行方法的信息(实际参数、局部变量等)
3. 栈属于线程私有,不能实现线程间的共享!
4. 栈的存储特性是“先进后出,后进先出”
5. 栈是由系统自动分配,速度快!栈是一个连续的内存空间!
堆的特点如下:
1. 堆用于存储创建好的对象和数组(数组也是对象)
2. JVM只有一个堆,被所有线程共享
3. 堆是一个不连续的内存空间,分配灵活,速度慢!
方法区(又叫静态区)特点如下:
1. JVM只有一个方法区,被所有线程共享!
2. 方法区实际也是堆,只是用于存储类、常量相关的信息!
3. 用来存放程序中永远是不变或唯一的内容。(类信息【Class对象】、静态变量、字符串常量等)
总结一下,栈是属于线程私有的,存储的是局部变量,而堆是线程共享的,其中存储的是对象,而方法区实际上也是堆,属于堆的一块,里面存储的是类的有关信息,如字节码信息,即代码,还有静态属性和方法,字符串常量。
或许就上面的特点你还不能很好的理解java内存机制,接下来通过一个例子来讲解一下:
package OOTest;
/**
* 分析JVM内存机制
* @author lwdragon
*
*/
public class Student {
String name;
Book book;
Student(String name,Book book){
this.name=name;
this.book=book;
}
void study(){
System.out.println(name+" 在学习 "+book.name);
}
void display(){ //返回book地址
System.out.println(book);
}
public static void main(String[] args) {
Book b=new Book("Java");
System.out.println("b的地址:"+b);
Student s=new Student("wesley",b);
s.study();
s.display();
}
}
class Book{
String name;
Book(String name){
this.name=name;
}
}
输出结果:
b的地址:OOTest.Book@15db9742
wesley 在学习 Java
OOTest.Book@15db9742
我们来仔细分析上面代码的内存机制。
栈 堆 方法区
当源代码被编译成字节码,即调用java Student时,jvm开始解释字节码,从main方法程序的入口开始,有关字符串常量即“Java”、“b的地址”、“wesley”、“在学习”和字节码存储在堆中的方法区内;
之后在栈中申请一个栈帧,里面存储着main方法的局部变量。当new book的时候,首先会在堆中申请一个内存空间,里面存储着属性,并赋初始值“”,之后调用该对象的构造方法,构造方法会在栈中再申请一个栈帧,存储则局部变量name,构造完成后,释放栈帧,此时,将b在堆中的内存地址即输出结果的15db9742赋给了在main方法中的b变量;
读到下一行代码,接下输出b的地址;
再读到下一行代码,当new Student()时,先在堆中申请一个空间,在未构造之前,name="",book=null,方法存储在方法区中的code-segment中。开始构造后,将方法区的“wesley”的内存传给name,b的值传给book,注意b 的值是一个内存地址,之后将这块空间的地址赋给main方法的s局部变量;
之后再读到下一行代码,调用s.study();
最后,调用s.display();
此时main中的所有代码执行完成,栈中的main栈帧释放。
------百战004天