前端编译:
使用前端编译期(javac、eclipse)将.java源代码编译成.class字节码文件,运行时JVM加载.class字节码文件交给解释器解释成相应平台的机器码执行
早期的JVM就是以这种方式运行java程序,由解释器一行行的将字节码解释成机器码
缺点:
对代码运行效率没有任何优化措施
一行行解释效率低下
JIT编译(即时编译)
通过收集“热点代码”使用JVM内置的即时编译器,在运行时将字节码编译成机器码,并且编译的机器码缓存在方法区中,大大提升执行效率。
缺点:
收集“热点代码”占用程序资源影响程序运行
编译本地代码过程占用程序运行时间
编译机器码缓存在方法区占用内存
另外虽然JIT编译器不是JVM必须部分,JVM也没要求JIT编译器必须存在,但是JIT编译器的编译速度和编译的代码质量却是衡量一个JVM性能的很重要的指标,也是最能体现虚拟机技术水平的部分。
JIT编译器与解释器的工作模式
可以通过“-version”参数显示当前的工作模式,各工作模式说明如下:
1、混合模式(Mixed Mode)
JIT编译器(无论C1还是C2)与解释器配合工作的方式;
这是默认的方式,也可通过“-Xmixed”参数设定;
2、解释模式(Interpreted Mode)
全部代码由解释器解释执行,JIT编译器不介入工作;
可以通过“-Xint”参数设定;
3、编译模式(Compiler Mode)
优先采用编译方式执行程序,但解释器仍要在编译无法时行时介入执行过程;
可以通过“-Xcomp”参数设定;
该参数强调的是首次调用方法时执行编译,并不是不用解释器;
一般情况下(不开启分层编译),一个方法需要解释执行一定次数后才编译
但JDK7/8作为默认开启分层编译策略;
分层编译
为了在程序启动响应速度与运行效率之间达到最佳平衡,会启用分层编译(Tiered Compilation)策略;
1、编译层次
根据编译器编译、优化的规模与耗时,划分出不同的编译层次,包括:
(I)、第0层
程序解释执行,解释器不开启性能监控功能(Profiling),可触发第1层编译;
(II)、第1层
也称为C1编译,将字节码编译为本地代码,进行简单、可靠的优化,如有必要加入性能监控的逻辑;
(III)、第2层
也称为C2编译,也是将字节码编译为本地代码,但进行一些编译耗时较长的优化,甚至会根据性能监控信息进行一些不可靠的激进优化;
2、优点
这时C1和C2同时进行工作,许多代码都可能被编译多次;
用C1获取更高的编译速度,用C2获取更好的编译质量;
解释器执行时也无须再承担收集性能监控信息的任务(如果不开启分层编译,又工作在Server模式,解释器提供监控信息给C2使用);
最终在程序启动响应速度与运行效率之间达到最佳平衡;