前言
做为一名Java程序员,咱们须要知道Java代码是怎么运行的。最近复习了深刻理解Java虚拟机这本书,作了一下笔记,但愿对你们有帮助,若是有不正确的地方,欢迎提出,感激涕零。前端
在这里小编建了一个前端学习交流扣扣群:132667127,我本身整理的最新的前端资料和高级开发教程,若是有想须要的,能够加群一块儿学习交流java
java 代码运行主要流程程序员
本文主要讲解流程以下:编程
java源文件编译为class字节码
类加载器把字节码加载到虚拟机的方法区。
运行时建立对象
方法调用,执行引擎解释为机器码
CPU执行指令
多线程切换上下文
编译安全
咱们都知道,java代码是运行在Java虚拟机上的。可是java是一门面向对象的高级语言,它不只语法很是复杂,抽象程度也很是高,并不能直接运行在计算机硬件机器上。数据结构
Java虚拟机(Java Virtual Machine 简称JVM)是运行全部Java程序的抽象计算机,是Java语言的运行环境。
所以,在运行Java程序以前,须要编译器把代码编译成java虚拟机所能识别的指令程序,这就是Java字节码,即class文件。多线程
因此,Java代码运行的第一步是:把Java源代码编译成.class 字节码文件。编程语言
类加载学习
在Class文件中描述的各类信息,须要被加载到虚拟机以后才能运行和使用。所以,须要把class字节码文件加载到Java虚拟机来。spa
虚拟机把描述类的数据从 Class 文件加载到内存,并对数据进行校验、转换解析和初始化,最终造成能够被虚拟机直接使用的 Java 类型,这就是虚拟机的
类加载机制。
加载
在加载阶段,虚拟机须要完成如下3件事情:
经过一个类的全限定名来获取定义此类的二进制字节流。
将这个字节流所表明的静态存储结构转化为方法区的运行时数据结构。
在内存中生成一个表明这个类的java.lang.Class对象,做为方法区这个类的各类数据的访问入口
加载阶段完成后,这些二进制字节流按照虚拟机所需的格式存储在方法区之中。
验证
为了确保Class文件的字节流中包含的信息符合当前虚拟机的要求,不会危害虚拟机的安全,Java虚拟机对输入的字节流走验证过程。
验证阶段包括四个阶段:文件格式验证、元数据验证、字节码验证、符号引用验证。
文件格式验证:验证字节流是否符合Class文件格式规范,如:是否以魔数0xCAFEBABE开头。
元数据验证:对字节码描述的信息进行语义分析,如:这个类的父类是否继承了不容许被继承的类(被final修饰的类);
字节码验证:主要目的是经过数据流和控制流分析,肯定程序语义是合法的、符合逻辑的。如:保证跳转指令不会跳转到方法体之外的字节码指令上。
符号引用验证:发生在虚拟机将符号引用转化为直接引用的时候,如:校验符号引用中经过字符串描述的全限定名是否能找到对应的类。
准备
准备阶段是正式为类变量分配内存并设置类变量初始值,这些变量所使用的内存都将在方法区中进行分配。如:
public static int value =123;
变量value在准备阶段事后的初始值是0而不是123。
解析
解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。
好比:com.User类引用com.Tool类,在编译时,User类不知道Tool类的实际内存地址,所以只能使用符号com.Tool(假设)来表示。而在类加载加载User类的时候,能够经过虚拟机获取Tool类的实际内存地址,所以即可以将符号com.Tool替换为Tool类的实际内存地址,即直接引用地址。
解析动做主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用点限定符 7 类符号引用进行。
初始化
到了初始化阶段,才真正开始执行类中定义的Java字节码。在这个阶段,则根据程序员经过程序制定的主观计划去初始化类变量和其余资源。
建立对象
Java虚拟机是如何执行字节码的呢?咱们先来看一下运行时建立对象。
Java是面向对象的编程语言,程序的运行是以对象为调用单位的。
字节码文件加载到虚拟机的方法区后,在程序运行过程,经过 class字节码文件建立与其对应的对象信息 。
建立对象的方式有:new关键字,反射等。
Java堆内存是线程共享的区域,建立后的对象信息就保存在Java堆内存中。
方法调用
JVM的调用单位是对象,可是真正执行功能性的代码仍是对象上的方法。
在运行过程当中,每当调用进入一个java方法,java虚拟机会在当前线程的java方法栈中生成一个栈帧,用以存放局部变量以及字节码的操做数。方法栈内存是线程私有的,每一个线程都有本身的方法栈。若是对应的方法是本地方法,则对应的就是本地方法栈。
java运行时数据区域以下:
解释
当调用Java对象的某个方法时,JVM执行引擎会将该方法的字节码文件翻译成计算机所能识别的机器码,机器码信息保存在方法区中。翻译有解释执行和即时编译两种方式。
两种翻译方式的区别以下:
解释执行来一行代码,解释一行,大部分不经常使用的代码,都是采用这种方式。
即便编译
对于部分热点代码,将一个方法包含的全部字节码翻译成机器指令,以提升java虚拟机的运行效率。
即时编译是创建经典的二八定律上,即20%代码占据了80%的计算资源。
执行指令
Java程序被加载入内存后,指令也在内存中了。
指令的指令寄存器IP,指向下一条待执行指令的地址。
CPU的控制单元根据IP寄存器的指向,将主存中的指令装载到指令寄存器,这些加载的指令就是一串二进制码,还须要译码器进行解码。
解码后,若是须要获取操做数,则从内存中取数据,调用运算单元进行计算。
多线程上下文切换
CPU一通上电,就会周而复始从内存中获取指令、译码、执行。
为了支持多任务,CPU 将执行时间这个资源划分红时间片,每一个程序执行一段时间。
java虚拟机的多线程是经过线程轮流切换分配处理执行时间的方式来实现的,在任何一个肯定的时刻,一个处理器(对于多核处理器来讲是一个内核)都只会执行一条程序中的指令。
假设当前线程在运行中,CPU分配的时间执行完了,总得保存运行过的结果信息吧,要否则白白浪费以前的工做了,所以,程序计数器(PC寄存器)做用体现出来了,它是一块较小的内存空间,线程私有,能够看做当前线程执行的字节码的行号指示器。当CPU又给它分配时间跑的时候,能够把数据恢复,接着上一次执行到的位置继续执行就能够了。