VM(HotSpot)执行引擎中包含解释器与JIT编译器
热点代码(执行多次)才有JIT编译的必要(JIT编译阈值)
JVM(HotSpot)会有两个计数器(次数/回边)判断方法/代码块是否为热点代码(达到JIT编译阈值)
JIT编译后会进行OSR(On Stack Replacement)栈上方法替换


什么是JIT

Just In Time,意为准时制作业,在编程世界里,狭义上指“即时编译”:某段代码即将第一次被执行时进行编译。但慢慢JIT被泛化,等价动态编译(dynamic compilation):在运行时进行编译;与之相对的是事前编译(ahead-of-time compilation,简称AOT),也叫静态编译(static compilation)。


前情提要:JVM运行原理

JVM运行原理

在部分商用虚拟机中(如HotSpot),Java程序刚开始执行是通过解释器(Interpreter)进行解释执行的,当虚拟机发现某个方法或代码块的运行特别频繁时,就会把这些代码认定为“热点代码”。为了提高热点代码的执行效率,在运行时,虚拟机将会把这些代码编译成与本地平台相关的机器码,并进行各种层次的优化,完成这个任务的编译器称为即时编译器(Just In Time Compiler,下文统称JIT编译器)。

即时编译器并不是虚拟机必须的部分,Java虚拟机规范并没有规定Java虚拟机内必须要有即时编译器存在,更没有限定或指导即时编译器应该如何去实现。但是,即时编译器编译性能的好坏、代码优化程度的高低却是衡量一款商用虚拟机优秀与否的最关键的指标之一,它也是虚拟机中最核心且最能体现虚拟机技术水平的部分。

由于Java虚拟机规范并没有具体的约束规则去限制即使编译器应该如何实现,所以这部分功能完全是与虚拟机具体实现相关的内容,如无特殊说明,我们提到的编译器、即时编译器都是指Hotspot虚拟机内的即时编译器,虚拟机也是特指HotSpot虚拟机。


为什么HotSpot虚拟机要使用解释器与编译器并存的架构?

尽管并不是所有的Java虚拟机都采用解释器与编译器并存的架构,但许多主流的商用虚拟机(如HotSpot),都同时包含解释器和编译器。

解释器与编译器两者各有优势:

解释器:
快速启动:当程序需要迅速启动和执行的时候,解释器可以首先发挥作用,省去编译的时间,立即执行;
节约内存:当程序运行环境中内存资源限制较大(如部分嵌入式系统中),可以使用解释器执行节约内存,反之可以使用编译执行来提升效率;
逆优化:此外,如果编译后出现“罕见陷阱”,可以通过逆优化退回到解释执行。
编译器:
效率优化:在程序运行后,随着时间的推移,编译器逐渐发挥作用,把越来越多的代码编译成本地代码之后,可以获取更高的执行效率。


JIT编译比解释快?

要解释器解释执行,抽象的看是这样的:

输入的代码 -> [ 解释器 解释执行 ] -> 执行结果

而要JIT编译执行的话,抽象的看则是:

输入的代码 -> [ 编译器 编译 ] -> 编译后的代码 -> [ 执行 ] -> 执行结果

说JIT比解释快,其实说的是执行编译后的代码比解释器解释执行要快,并不是说编译这个动作比解释这个动作快。

JIT编译再怎么快,至少也比解释执行一次略慢一些,而要得到最后的执行结果还得再经过一个“执行编译后的代码”的过程。

所以,对“只执行一次”的代码而言,解释执行其实总是比JIT编译执行要快。

怎么算是“只执行一次的代码”呢?粗略说,下面两个条件同时满足时就是严格的“只执行一次”

1、只被调用一次,例如类的构造器(class initializer,<clinit>())

2、没有循环

对只执行一次的代码做JIT编译再执行,可以说是得不偿失。
对只执行少量次数的代码,JIT编译带来的执行速度的提升也未必能抵消掉最初编译带来的开销。
只有对频繁执行的代码,JIT编译才能保证有正面的收益。


哪些程序代码会被编译为本地代码?如何编译为本地代码?

程序中的代码只有是热点代码时,才会编译为本地代码,那么什么是热点代码呢?

运行过程中会被即时编译器编译的“热点代码”有两类:

被多次调用的方法。
被多次执行的循环体。

两种情况,编译器都是以整个方法作为编译对象。 这种编译方法因为编译发生在方法执行过程之中,因此形象的称之为栈上替换(On Stack Replacement,OSR),即方法栈帧还在栈上,方法就被替换了。


编译的空间开销

对一般的Java方法而言,编译后代码的大小相对于字节码的大小,膨胀比达到10x是很正常的。同上面说的时间开销一样,这里的空间开销也是,只有对执行频繁的代码才值得编译,如果把所有代码都编译则会显著增加代码所占空间,导致“代码爆炸”。

这也就解释了为什么有些JVM会选择不总是做JIT编译,而是选择用解释器+JIT编译器的混合执行引擎。


为何HotSpot虚拟机要实现两个不同的JIT即时编译器?

HotSpot虚拟机中内置了两个即时编译器:Client Complier和Server Complier,简称为C1、C2编译器,分别用在客户端和服务端。目前主流的HotSpot虚拟机中默认是采用解释器与其中一个编译器直接配合的方式工作。

程序使用哪个编译器,取决于虚拟机运行的模式。HotSpot虚拟机会根据自身版本与宿主机器的硬件性能自动选择运行模式,用户也可以使用“-client”或“-server”参数去强制指定虚拟机运行在Client模式或Server模式。
用Client Complier获取更高的编译速度,用Server Complier 来获取更好的编译质量。为什么提供多个即时编译器与为什么提供多个垃圾收集器类似,都是为了适应不同的应用场景。


Server Compiler与Client Compiler

Server Compiler和Client Compiler两个编译器的编译过程是不一样的。

对Client Compiler来说,它是一个简单快速的编译器,主要关注点在于局部优化,而放弃许多耗时较长的全局优化手段。
而Server Compiler则是专门面向服务器端的,并为服务端的性能配置特别调整过的编译器,是一个充分优化 过的高级编译器。


如何判断方法或一段代码是不是热点代码呢?

要知道方法或一段代码是不是热点代码,是不是需要触发JIT即时编译,需要进行Hot Spot Detection(热点探测)。

目前主要的热点探测方式有以下两种:

基于采样 的热点探测
采用这种方法的虚拟机会周期性地检查各个线程的栈顶,如果发现某些方法经常出现在栈顶,那这个方法就是“热点方法”。
这种探测方法的好处是实现简单高效,还可以很容易地获取方法调用关系(将调用堆栈展开即可),缺点是很难精确地确认一个方法的热度,容易因为受到线程阻塞或别的外界因素的影响而扰乱热点探测。
基于计数器 的热点探测
采用这种方法的虚拟机会为每个方法(甚至是代码块)建立计数器,统计方法的执行次数,如果执行次数超过一定的阀值,就认为它是“热点方法”。
这种统计方法实现复杂一些,需要为每个方法建立并维护计数器,而且不能直接获取到方法的调用关系,但是它的统计结果相对更加精确严谨。


HotSpot虚拟机中使用的是哪钟热点检测方式呢?

在HotSpot虚拟机中使用的是第二种—— 基于计数器的热点探测方法 ,因此它为每个方法准备了两个计数器:方法调用计数器和回边计数器。在确定虚拟机运行参数的前提下,这两个计数器都有一个确定的阈值,当计数器超过阈值溢出了,就会触发JIT编译。

方法调用计数器

顾名思义,这个计数器用于统计方法被调用的次数。

当一个方法被调用时,会先检查该方法是否存在被JIT编译过的版本。如果存在,则优先使用编译后的本地代码来执行;如果不存在已被编译过的版本,则将此方法的调用计数器值加1。
然后判断方法调用计数器与回边计数器值之和是否超过方法调用计数器的阈值。如果超过阈值,那么将会向即时编译器提交一个该方法的代码编译请求。
如果不做任何设置,执行引擎并不会同步等待编译请求完成,而是继续进行解释器按照解释方式执行字节码,直到提交的请求被编译器编译完成。
当编译工作完成之后,这个方法的调用入口地址就会系统自动改写成新的,下一次调用该方法时就会使用已编译的版本。

回边计数器

它的作用就是统计一个方法中循环体代码执行的次数,在字节码中遇到控制流向后跳转的指令称为“回边 ”。

JIT-即时编译技术相关推荐

  1. JIT 即时编译及优化技术

    JIT 即时编译及优化技术 前言 即时编译 热点代码探测 编译优化技术 语言无关的经典优化技术之一:公共子表达式消除 语言相关的经典优化技术之一:数组范围检查消除 最重要的优化技术之一:方法内联 最前 ...

  2. javac参数 编译警告关闭_JVM之JIT即时编译

    当今Java语言被广为接受的优点之一就有即时编译,即时编译的存在使得Java应用可以运行时间的增长而获得更高的性能. 如果有对jvm做过研究的朋友,一定听说过这样一段话:由于即时编译技术的进步,尤其是 ...

  3. (三)JVM成神路之全面详解执行引擎子系统与JIT即时编译原理

    引言 执行引擎子系统是JVM的重要组成部分之一,在JVM系列的开篇曾提到:JVM是一个架构在平台上的平台,虚拟机是一个相似于"物理机"的概念,与物理机一样,都具备代码执行的能力.但 ...

  4. JVM成神路之全面详解执行引擎子系统、JIT即时编译原理与分派实现

    引言 执行引擎子系统是JVM的重要组成部分之一,在JVM系列的开篇曾提到:JVM是一个架构在平台上的平台,虚拟机是一个相似于"物理机"的概念,与物理机一样,都具备代码执行的能力.但 ...

  5. jit即时编译_热点中的即时编译器(JIT)

    jit即时编译 即时编译器(JIT)的概念以及更广泛的自适应优化是除Java(.Net,Lua,JRuby)之外的许多语言中众所周知的概念. 为了解释什么是JIT编译器,我想先定义一个编译器概念. 根 ...

  6. 【Flutter】Dart 语言 ( Dart 语言特征 | JIT 即时编译 | AOT 静态编译 )

    文章目录 I . Dart 语言特征 II . JIT 即时编译 III . AOT 静态编译 I . Dart 语言特征 1 . Dart 语言特征 : ① 语言类型 : 强类型语言 ; ② 编译类 ...

  7. JVM 虚拟机 AOT 和 JIT 即时编译

    文章目录 JVM 虚拟机 AOT 和 JIT 即时编译 1.AOT 2.JIT JVM 虚拟机 AOT 和 JIT 即时编译 1.AOT AOT 是 Ahead Of Time ,运行前编译 与 JI ...

  8. JVM(1)之JVM的组成详解(字符串常量池+双亲委派机制+JIT即时编译......)

    以下总结自:<深入理解java虚拟机> + 宋红康老师视频 字节码文件介绍:深入理解JVM之Java字节码(.class)文件详解_Windy_729的博客-CSDN博客_字节码文件 JV ...

  9. JIT - 即时编译

    即时编译 即时编译(英语:Just-in-time compilation),又译及时编译.实时编译[3],动态编译的一种形式,是一种提高程序运行效率的方法.通常,程序有两种运行方式:静态编译与動態直 ...

  10. java jit 即时编译_JVM即时编译(JIT)

    版权声明:本文为CSDN博主「nogos」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明. 原文链接:https://blog.csdn.net/sunxiang ...

最新文章

  1. 一次内网靶场学习记录
  2. VTK修炼之道44:图形进阶_vtkPolyData数据源讨论与数据创建
  3. 开源|如何开发一个高性能的redis cluster proxy?
  4. 看了这一篇,就不用看别的——Java中Object关于锁的的三个方法:wait,notify,notifyAll的作用
  5. 12人类为什么有战争
  6. 【计算机视觉】运动目标检测算法文献阅读笔记
  7. mysql递归查询所有子级,完整PDF
  8. 简单介绍.Net3.0 中跨线程访问控件
  9. Liferay中页面的权限控制
  10. vuex的基础小案例(黑马教程)
  11. android 人脸识别边框_Android相机预览,指定区域显示预览框,在区域内出现人脸进行人脸识别,并抓拍人脸照片存在本地,CameraX,虹软人脸识别...
  12. TCP长连接开发相关,调试工具SocketTool与框架GatewayWorker
  13. 性能优化专题 - MySql 性能优化 - 03 - 深入理解InnoDB
  14. 和秋叶一起学PPT之线条(课时六)
  15. DELMIA软件:机器人与行走轴联动实现长焊缝弧焊焊接虚拟仿真
  16. 阿里巴巴矢量图标如何修改颜色,以及原理
  17. SQL SERVER SA密码忘记,windows集成身份验证都登录不了不怎么办
  18. Python获取指定月份的所有天数
  19. 从一个置顶CPT广告学习计算广告设计要点
  20. win7 蓝牙4.0 ble驱动_恩智浦发布行业领先的2x2 Wi-Fi 6 +蓝牙解决方案,彻底改变游戏、音频、工业和物联网市场...

热门文章

  1. 《神探狄仁杰》主题曲《长歌一曲》
  2. 30天全网2W粉,感谢坚持!
  3. Android 11.0 设置Camera2的相机拍照默认像素为1080P
  4. 基于深度学习的实现影像地图道路提取
  5. 第十一届蓝桥软件类JavaB组
  6. Windows下,Pytorch使用Imagenet-1K训练ResNet的经验(有代码)
  7. 阡陌路-车行天下之汽车基础知识
  8. Ubuntu 下CodeLite 的安装
  9. 机械臂关节模组制动相关(零差云控eBr)
  10. 高并发到底要怎么算才是高并发?