JAVA的JVM的内存可分为3个区:堆(heap)、栈(stack)和方法区(method)java
堆区:
1.存储的所有是对象,每一个对象都包含一个与之对应的class的信息。(class的目的是获得操做指令)
2.jvm只有一个堆区(heap)被全部线程共享,堆中不存放基本类型和对象引用,只存放对象自己.
3.通常由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。程序员
栈区:
1.每一个线程包含一个栈区,栈中只保存基础数据类型的对象和自定义对象的引用(不是对象),对象都存放在堆区中
2.每一个栈中的数据(原始类型和对象引用)都是私有的,其余栈不能访问。
3.栈分为3个部分:基本类型变量区、执行环境上下文、操做指令区(存放操做指令)。
4.由编译器自动分配释放 ,存放函数的参数值,局部变量的值等.jvm
静态区/方法区:
1.方法区又叫静态区,跟堆同样,被全部的线程共享。方法区包含全部的class和static变量。
2.方法区中包含的都是在整个程序中永远惟一的元素,如class,static变量。
3.—,全局变量和静态变量的存储是放在一块的,初始化的全局变量和静态变量在一块区域, 未初始化的全局变量和未初始化的静态变量在相邻的另外一块区域。函数
实例:测试
public class Sample // 运行时, jvm 把Sample的信息都放入方法区 { /** 范例名称 */ private String name; // new Sample实例后, name 引用放入栈区里, name 对象放入堆里 /** 构造方法 */ public Sample(String name) { this.name = name; } /** 输出 */ public void printName() { // print方法自己放入 方法区里。 System.out.println(name); } public static void main(String[] args) { // main 方法自己放入方法区。 Sample test1 = new Sample(" 测试1 "); // test1是引用,因此放到栈区里,new Sample(" 测试1 ")放入到堆区 test1.printName(); } }
Sample test1=new Sample("测试1");this
语句很简单啦,就是让java虚拟机建立一个Sample实例,而且呢,使引用变量test1引用这个实例。貌似小case一桩哦,就让咱们来跟踪一下Java虚拟机,看看它到底是怎么来执行这个任务的:spa
一、 Java虚拟机一看,不就是创建一个Sample实例吗,简单,因而就直奔方法区而去,先找到Sample类的类型信息再说。结果呢,嘿嘿,没找到@@,这会儿的方法区里尚未Sample类呢。可Java虚拟机也不是一根筋的笨蛋,因而,它发扬“本身动手,丰衣足食”的做风,立马加载了Sample类,把Sample类的类型信息存放在方法区里。线程
二、 好啦,资料找到了,下面就开始干活啦。Java虚拟机作的第一件事情就是在堆区中为一个新的Sample实例分配内存, 这个Sample实例持有着指向方法区的Sample类的类型信息的引用。这里所说的引用,实际上指的是Sample类的类型信息在方法区中的内存地址,其实,就是有点相似于C语言里的指针啦~~,而这个地址呢,就存放了在Sample实例的数据区里。指针
三、 在JAVA虚拟机进程中,每一个线程都会拥有一个方法调用栈,用来跟踪线程运行中一系列的方法调用过程,栈中的每个元素就被称为栈帧,每当线程调用一个方法的时候就会向方法栈压入一个新帧。这里的帧用来存储方法的参数、局部变量和运算过程当中的临时数据。OK,原理讲完了,就让咱们来继续咱们的跟踪行动!位于“=”前的Test1是一个在main()方法中定义的变量,可见,它是一个局部变量,所以,它被会添加到了执行main()方法的主线程的JAVA方法调用栈中。而“=”将把这个test1变量指向堆区中的Sample实例,也就是说,它持有指向Sample实例的引用。code
OK,到这里为止呢,JAVA虚拟机就完成了这个简单语句的执行任务。参考咱们的行动向导图,咱们终于初步摸清了JAVA虚拟机的一点点底细了,COOL!接下来,JAVA虚拟机将继续执行后续指令,在堆区里继续建立另外一个Sample实例,而后依次执行它们的printName()方法。当JAVA虚拟机执行test1.printName()方法时,JAVA虚拟机根据局部变量test1持有的引用,定位到堆区中的Sample实例,再根据Sample实例持有的引用,定位到方法去中Sample类的类型信息,从而得到printName()方法的字节码,接着执行printName()方法包含的指令