简单过程:源文件经过编译成二进制字节码文件保存到磁盘,jvm读取字节码文件经过类加载器加载,然后经过解释器解释成各个系统能够读懂执行二进制机器码。
一,编译:
Java源文件(.java)通过JDK的java编译器(javac.exe)编译成二进制字节码文件(.class)保存到硬盘。(以maven项目为例,经过编译后产生jar包或者war包保存到target目录下)
编译后的字节码文件格式主要分为两部分:常量池和方法字节码。
常量池:记录的是代码出现过的字面量(文本字符串、八种基本类型的值、被声明为final的常量等)以及符号引用(类和方法的全限定名、字段的名称和描述符、方法的名称和描述符);
https://www.cnblogs.com/superyc/p/9975254.html
方法字节码: 中放的是各个方法的字节码(依赖操作数栈和局部变量表,由JVM解释执行)
二,运行:
1,类的加载
(加载->连接->初始化)
连接包括(验证->准备->解析)
加载
1, 通过全类名获取定义此类的二进制字节流(从zip,jar,ear,war等获取)
2,将字节流所代表的静态存储结构转换为方法区的运行时数据结构
3,在内存中生成一个代表该类的 Class 对象(java.Lang.Class),作为方法区这些数据的访问入口
准备
准备阶段是正式为类变量分配内存并设置类变量初始值的阶段,这些内存都将在方法区中分配。
- 这时候进行内存分配的仅包括类变量(static),而不包括实例变量,实例变量会在对象实例化时随着对象一块分配在 Java 堆中。
- 这里所设置的初始值"通常情况"下是数据类型默认的零值(如0、0L、null、false等),比如我们定义了public static int value=111 ,那么 value 变量在准备阶段的初始值就是 0 而不是111(初始化阶段才会复制)。特殊情况:比如给 value 变量加上了 fianl 关键字public static final int value=111 ,那么准备阶段 value 的值就被复制为 111。
解析
解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。解析动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用限定符7类符号引用进行。
初始化
初始化是类加载的最后一步,也是真正执行类中定义的 Java 程序代码(字节码),初始化阶段是执行类构造器 <clinit> ()方法的过程。(通俗点就是初始化实例对象,给对象分配内存)
public class Main {
public static void main(String[] args) {
Animal animal = new Animal("Tom");
animal.printName();
}
}
class Animal{
private String name;
public Animal(String name) {
super();
this.name = name;
}
public void printName(){
System.out.println("Animal = " + this.name);
}
}
具体实例(Test.java表示当前类)(Anmimal表示class类对象)(animal 表示类的对象,实例)
(1)在类路径下找到编译好的 java 程序中得到 Test.class 字节码文件后,在命令行上敲 java Test,系统就会启动一个 JVM 进程,JVM进程从classpath路径下找到一个名为Test.class的二进制文件,将Test.class文件中的 类信息加载到运行时数据区的方法区(JDK 8 方法区存在 堆区) 中,这一过程叫做类的加载。(加载)(只有类信息在方法区中,才能创建对象,使用类中的成员变量);
(2)JVM 找到main方法的主函数入口, 持有一个指向当前类(Test)常量池的指针,而常量池中的第一项发现是一个对Animal对象的符号引用,并且main方法中第一条指令是Animal animal = new Animal("Tom"),就是让JVM创建一个Animal类的实例,但是方法区中还没有Animal类的类信息,于是JVM就要马上加载编译好Animal.class,将Animal类信息放入到方法区中,并且产生了一个Anmimal类对象,于是JVM 以一个直接指向方法区 Anmimal类对象的指针(直接引用)替换了常量池中第一项的符号引用。(解析)
(3)加载完Animal类的信息以后,JVM虚拟机就会在堆内存中为一个Anmimal类对象的实例animal分配内存,然后调用其构造函数初始化animal实例,(初始化)这个实例持有指向方法区的Anmimal类对象的类型信息(其中包含有方法表,java动态绑定的底层实现)的引用。(animal指向了Anmimal类对象的引用会自动的放在栈中,字符串常量"Tom"会自动的放在方法区的运行时常量池中,对象会自动的放入堆区)
(4)当使用 animal.pringName()的时候,JVM根据栈中animal引用找到Anmimal类对象,然后根据Animal类对象持有的引用定位到方法区中Animal类的类型信息方法表,获得pringName()函数的字节码地址,然后Java虚拟机执行引擎依赖局部变量表,操作数栈进行字节码解释执行,(解释执行)返回结果!