给Java字节码加上”翅膀“的JIT编译器
Posted 我是攻城师
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了给Java字节码加上”翅膀“的JIT编译器相关的知识,希望对你有一定的参考价值。
上面文章在介绍Java的内存模型的时候,提到过由于编译器的优化会导致重排序的问题,其中一个比较重要的点地方就是关于JIT编译器的功能。JIT的英文单词是Just In Time翻译成中文就是及时,恰好的意思,意在说明JIT编译器优化java的class文件里面的byte code是拿捏的恰到好处。
JIT编译器是JRE里面的一个为了在运行时提升Java程序性能的一个重要组件,我们知道Java代码一大优势就是在于一次编写,到处运行的特点。Java程序通常在编译后是一大堆class文件,也就是我们所说的字节码,然后通过JVM来解释执行这些与平台无关的字节码,从而屏蔽了操作系统的差异,做到了跨平台的特点。但jvm在在运行时候执行class文件的字节码的时候性能并没有执行跟操作系统直接有关的机器指令性能来的快,正是因为这个原因,才出现JIT编译器,目的就是为了提高执行效率。
到这里,我们再总结下JIT编译器的定义:
在Java程序运行时把一些class文件的字节码给转变成操作系统本地的指令码,从而提升程序性能。
如下图
在上面的图我们能够看到,我们的java源文件先在编译时被转成class字节码文件,然后在运行时会在当一个方法第一次调用时会被JIT再次编译优化转成native machine code也就是上面说的操作系统级别的指令。
这里面大家注意到仅仅当方法第一次调用时才会进行JIT优化,那么有个问题是既然JIT编译器优化运行时执行性能,为啥不把所有的方法都优化一次呢,而非得时用到的时候才优化呢? 这正是just in time写照。 当JVM启动时候,实际上是会加载数千个方法的,理论上把所有方法都提前通过JIT转换一下是会提升更多运行性能,但实际情况是JIT编译优化是需要耗费一定的cpu和内存资源(用来缓存指令),这也意味着如果直接优化所有方法,有可能导致jvm启动的非常慢,即使它能在运行时带来的一定的性能提升。
此外,JIT在运行时做编译优化是需要重新理解字节码的语义的,为了分析方法,它的字节码会被转成一种叫做trace tree的数据结构,然后根据方法的trace tree来做相关分析和优化,最终字节码会被转成本地的机器码。当然JIT会使用多线程来编译task,从而使java应用程序也就是jvm启动的更快。到这里相信看了我上篇文章的同学们,就会明白重排序的问题。
你看到的代码顺序,未必是其执行顺序
世界如此复杂 你要学会装傻
以上是关于给Java字节码加上”翅膀“的JIT编译器的主要内容,如果未能解决你的问题,请参考以下文章