在部分商用虚拟机中,Java程序最初是经过解释器(Interpreter)进行解释执行的,当虚拟机发现某个方法或者代码块的运行特别频繁时,就会把这些代码认定为“热点代码”(Hot Spot Code)。为提升热点代码的执行效率,在运行时,虚拟机将会把这些代码编译成与本地平台相关的机器码,并进行这种层次的优化,完成这个任务的编译器成为即时编译器(Just In Time Compiler,简称JIT编译器)。架构
Java虚拟机规范并无规定Java虚拟机必需要有即时编译器存在。可是,即时编译器编译性能的好坏、代码优化程度的高低确实衡量一款商用虚拟机优秀与否的关键指标之一,也是虚拟机中最核心且能体现虚拟机技术水平的部分。性能
HotSpot虚拟机内的即时编译器优化
将要了解HotSpot虚拟机内即时编译器的运做过程,同时,还要解决一下几个问题:线程
为什么Hotspot虚拟机要使用解释器与编译器并存的架构?对象
为什么Hotspot虚拟机要实现两个不一样的即时编译器?blog
程序什么时候使用解释器执行?什么时候使用编译器执行?继承
那些程序代码会被编译为本地代码?如何编译为本地代码?内存
如何从外部观察即时编译器的编译过程和编译结果?资源
解释器与编译器编译器
解释器与编译器二者各有优点:当程序须要迅速启动和执行的时候,解释器能够首先发挥做用,省去编译的时间,当即执行。在程序运行后,随着时间的推移,解释器逐渐发挥做用,把愈来愈多的代码编译成本地代码以后,能够获取更高的执行效率。当程序运行环境的内存资源有限制较大(如部分嵌入式系统),可使用解释器执行节约内存,反之可使用编译器执行来提高效率。同时,解释器还能够做为编译器激进优化时的一个“逃生门”,让编译器根据几率选择一些大多数时都能提高运行速度的优化手段,当激进优化的假设不成立,如加载了新类后类型继承结构出现变化、出现“罕见陷阱”(Uncommon Trap)时能够经过逆优化(Deoptimization)退回到解释状态继续执行,所以,整个虚拟机执行架构中,解释器与编译器常常配合工做。
Hotspot虚拟机中内置了两个即时编译器,分别为Client Compiler和Server Compiler,或者简称C1编译器和C2编译器。目前主流的Hotspot虚拟机中,默认采用解释器与其中一个编译器直接配合的方式工做,程序使用那个编译器,取决于虚拟机运行的模式,Hotspot虚拟机会根据自身版本与宿主机器的硬件性能自动选择运行模式,用户也可设置参数强制虚拟机使用的模式。
编译器和解释器搭配使用的方式成为“混合模式”(Mixed Mode),用户可设置参数强制虚拟机使用其中一种。
因为即时编译器编译本地代码须要占用程序运行时间,要编译出优化程度更高的代码,所花费的时间可能更长;并且想要编译出优化程度更高的代码,解释器可能还要替编译器作收集性能监控信息,这对解释执行的速度也有影响。为了在程序启动响应速度与运行效率之间达到最佳平衡,Hotspot虚拟机还会逐渐启用分层编译(Tiered Compilation)的策略。分层编译根据编译器编译、优化的规模与耗时,划分出不一样编译层次:
第0层,程序解释执行,解释器不开启性能监控功能,可触发第1层编译。
第1层,也称C1编译,将字节码编译为本地代码,进行简单、可靠的优化,若有必要将加入性能监控的逻辑。
第2层(或2层以上),也称C2编译,将字节码编译为本地代码,可是会启用一些编译耗时较长的优化,甚至会根据性能监控信息进行一些不可靠的激进优化。
实施分层编译后,Client Compiler和Server Compiler将会同时工做,许多代码均可能会被屡次编译,用Client Compiler获取更高的编译速度,用Server Compiler来获取更好的编译执行,在解释执行的时候也无须再承担收集性能监控信息的任务。
编译对象与触发条件
在运行中被即时编译器编译的“热点代码”有两类:
被屡次调用的方法。
被屡次执行的循环体。
判断一段代码是否是热点代码,是否是须要触发即时编译,这样的行为称为热点探测(Hot Spot Detection),进行热点探测并不必定要知道具体被调用了多少次,目前主要的热点探测断定方式有两种:
基于采样的热点探测(Sample Based Hot Spot Detection):采用这种方法的虚拟机会周期性地检查各个线程的栈顶,若是发现某个(或某些)方法常常出如今栈顶,那这个方法就是“热点方法”。基于采样的热点探测的好处是实现简单、高效,还能够很容易地获取方法调用关系(将调用堆栈展开便可),缺点是很难精确地确认一个方法的热度,容易由于受到线程阻塞或别的外界因素的影响而扰乱热点探测。
基于计数的热点探测(Counter Based Hot Spot Detection):采用这种方法的虚拟机会为每一个方法(甚至代码块)创建计数器,统计方法的执行次数,若是执行次数超过必定的阈值就认为它是“热点方法”。这种统计方法实现起来麻烦,须要为每一个方法维护计数器,并且不能直接获取到方法的调用关系,可是它的统计结果相对精确。
Hotspot虚拟机使用第二种。
编译过程
查看及分析即便编译结果
编译优化技术