本讲将介绍Java代码是如何一步步运行起来的,其中涉及的编译器,类加载器,字节码校验器,解释器和JIT编译器在整个过程当中是发挥着怎样的做用。此外还会介绍Java程序所占用的内存是被如何管理的:堆、栈和方法区都各自负责存储哪些内容。最后用一小块代码示例来帮助理解Java程序运行时内存的变化。java
细心的读者可能注意到了,在流程图中还涉及到一个叫JIT的东西在步骤中没有被解释。那么JIT编译器(Just-In-Time Compiler)是若是参与进程序的执行过程当中呢?让咱们来看如下两个例子。缓存
固然,这只是JIT编译器的优化手段之一,不一样公司设计的JIT编译器对Java程序的运行会有不一样的优化方式。此外须要知道的是,JIT编译器并非每次都会参与到执行过程当中来。jvm
在步骤3中咱们谈到字节码会被类加载器载入到内存,那么载入以后JVM是如何对其进行内存管理的呢?优化
一般,在载入内存后,一个Java程序所占用的内存会被大体分为3块区域:堆(heap),栈(stack)和方法区(method area)。操作系统
堆:存放new出来的东西。.net
栈:存放局部变量。线程
方法区:类型信息,字段信息,常量池(constant pool),静态变量,方法信息等。翻译
public final class Student extends Object implements Serializable { // 1.类信息 // 2.对象字段信息 private String name; private int score; // 3.常量池 public final int id = 0; public final String gender = "male"; // 4.静态变量 public static int a = 0; // 5.方法信息 public int getid() { return id; } }
PC寄存器:存放将要执行的指令的地址。(由于机器的脑子不灵活,因此须要一块专门的区域帮他记住执行到哪一步,否则它会忘记)设计
本地方法栈:与JVM栈所发挥的做用是很是类似的,其区别不过是JVM栈为Java方法服务,而本地方法栈则是为使用到的Native方法服务。有的虚拟机(例如Sun HotSpot虚拟机)甚至直接就把本地方法栈和虚拟机栈合二为一。code
每一个线程拥有各自独立的(虚拟机)栈、PC寄存器和本地方法栈。而堆和方法区则是全部线程共享的。
最后让咱们经过一个小例子来理解Java程序执行时内存的变化。
public class Person { int id; int age; Person(int id1, int age1) { id = id1; age = age1; } public static void main(String[] args) { Person Tom = new Person(1, 25); } }
首先,在stack中申请了一块内存,这块内存区域名字叫Tom,此时区域里存储的内容为null。
接着,调用Person的构造方法,方法的参数属于局部变量,所以在stack中有两块区域分别存放id1和age1。
经过构造方法,能够new出来一个Person的对象,这个对象连带着其成员变量会被存放在heap中。成员变量id和age的值由存放在stack中的局部变量id1和age1赋予。
最后,将这个对象的引用值(相似于地址)传递给Tom,经过引用值咱们就能够找到这个对象。
(注意:位于stack中的id1和age1会随着构造方法调用的结束而消失,这里为了更好地表现全过程,所以保留在图中。)
有问题欢迎你们在评论区留言,转载请注明出处。