JVM中的编译器

JVM中集成了两种编译器,Client Compiler和Server Compiler,它们的作用也不同。Client Compiler注重启动速度和局部的优化,Server Compiler则更加关注全局的优化,性能会更好,但由于会进行更多的全局分析,所以启动速度会变慢。两种编译器有着不同的应用场景,在虚拟机中同时发挥作用。

Client Compiler

HotSpot VM带有一个Client Compiler C1编译器。这种编译器启动速度快,但是性能比较Server Compiler来说会差一些。C1会做三件事:

  • 局部简单可靠的优化,比如字节码上进行的一些基础优化,方法内联、常量传播等,放弃许多耗时较长的全局优化。
  • 将字节码构造成高级中间表示(High-level Intermediate Representation,以下称为HIR),HIR与平台无关,通常采用图结构,更适合JVM对程序进行优化。
  • 最后将HIR转换成低级中间表示(Low-level Intermediate Representation,以下称为LIR),在LIR的基础上会进行寄存器分配、窥孔优化(局部的优化方式,编译器在一个基本块或者多个基本块中,针对已经生成的代码,结合CPU自己指令的特点,通过一些认为可能带来性能提升的转换规则或者通过整体的分析,进行指令转换,来提升代码性能)等操作,最终生成机器码。

Server Compiler

Server Compiler主要关注一些编译耗时较长的全局优化,甚至会还会根据程序运行的信息进行一些不可靠的激进优化。这种编译器的启动时间长,适用于长时间运行的后台程序,它的性能通常比Client Compiler高30%以上。目前,Hotspot虚拟机中使用的Server Compiler有两种:C2和Graal。

C2 Compiler

在Hotspot VM中,默认的Server Compiler是C2编译器。

C2编译器在进行编译优化时,会使用一种控制流与数据流结合的图数据结构,称为Ideal Graph。 Ideal Graph表示当前程序的数据流向和指令间的依赖关系,依靠这种图结构,某些优化步骤(尤其是涉及浮动代码块的那些优化步骤)变得不那么复杂。

Ideal Graph的构建是在解析字节码的时候,根据字节码中的指令向一个空的Graph中添加节点,Graph中的节点通常对应一个指令块,每个指令块包含多条相关联的指令,JVM会利用一些优化技术对这些指令进行优化,比如Global Value Numbering、常量折叠等,解析结束后,还会进行一些死代码剔除的操作。生成Ideal Graph后,会在这个基础上结合收集的程序运行信息来进行一些全局的优化,这个阶段如果JVM判断此时没有全局优化的必要,就会跳过这部分优化。

无论是否进行全局优化,Ideal Graph都会被转化为一种更接近机器层面的MachNode Graph,最后编译的机器码就是从MachNode Graph中得的,生成机器码前还会有一些包括寄存器分配、窥孔优化等操作。关于Ideal Graph和各种全局的优化手段会在后面的章节详细介绍。Server Compiler编译优化的过程如下图所示:

JVM中的编译器

Graal Compiler

从JDK 9开始,Hotspot VM中集成了一种新的Server Compiler,Graal编译器。相比C2编译器,Graal有这样几种关键特性:

  • 前文有提到,JVM会在解释执行的时候收集程序运行的各种信息,然后编译器会根据这些信息进行一些基于预测的激进优化,比如分支预测,根据程序不同分支的运行概率,选择性地编译一些概率较大的分支。Graal比C2更加青睐这种优化,所以Graal的峰值性能通常要比C2更好。
  • 使用Java编写,对于Java语言,尤其是新特性,比如Lambda、Stream等更加友好。
  • 更深层次的优化,比如虚函数的内联、部分逃逸分析等。

Graal编译器可以通过Java虚拟机参数-XX:+UnlockExperimentalVMOptions -XX:+UseJVMCICompiler启用。当启用时,它将替换掉HotSpot中的C2编译器,并响应原本由C2负责的编译请求。

 

参考原文:基本功 | Java即时编译器原理解析及实践 - 美团技术团队 (meituan.com)

 

原文链接:https://www.cnblogs.com/FangwayLee/p/17338548.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:JVM中的编译器 - Python技术站

(0)
上一篇 2023年4月22日
下一篇 2023年4月22日

相关文章

  • Java中OAuth2.0第三方授权原理与实战

    Java中OAuth2.0第三方授权原理与实战 OAuth2.0是一种基于授权的开放网络协议,用于安全地授权访问HTTP资源。 OAuth2.0第三方授权原理 OAuth2.0第三方授权过程主要涉及四个角色: 用户(User):需要获得第三方资源的用户 第三方应用(Client):需要调用第三方API的应用程序 第三方资源拥有者(Resource Owner…

    Java 2023年5月20日
    00
  • Java有趣好玩的图形界面开发八个案例实现

    感谢您对Java图形界面开发的关注,关于Java有趣好玩的图形界面开发八个案例实现的攻略,下面给你一个详细的介绍。 八个案例 Java图形界面开发八个案例是由Java研究社推出的一门Java课程。该课程主要通过八个实例,让大家初步掌握Java图形界面的开发。这八个案例包括:电子时钟、文字跑马灯、图片浏览器、字母连线游戏、翻牌游戏、俄罗斯方块、英雄连连看、飞机…

    Java 2023年5月18日
    00
  • SpringMVC程序简单实例

    SpringMVC程序简单实例 SpringMVC是一种基于Java的Web框架,它提供了许多便捷的功能和工具,使得开发者可以更加高效地开发Web应用程序。本文将详细讲解如何使用SpringMVC创建一个简单的Web应用程序,并提供两个示例来说明如何实现这一过程。 步骤一:创建SpringMVC项目 在开始使用SpringMVC创建Web应用程序之前,我们需…

    Java 2023年5月17日
    00
  • Java事务管理学习之Spring和Hibernate详解

    Java事务管理学习之Spring和Hibernate详解 什么是事务 事务是一组要么全部执行,要么全部不执行的操作。事务具有ACID(原子性、一致性、隔离性、持久性)特性。 在数据库中,事务可以保证数据的完整性和一致性。 Spring事务管理 Spring框架提供了对事务管理的支持。Spring事务管理是通过AOP实现的,在方法被调用时将事务逻辑织入到方法…

    Java 2023年5月20日
    00
  • Java中super关键字介绍以及super()的使用

    当子类需要引用父类的构造方法、成员变量或成员方法时,需要使用Java中的super关键字。super也可以理解为是当前对象的父类对象。 super的使用有以下几种形式: 使用super引用父类的成员变量和成员方法 在子类中可以使用super关键字来引用父类的成员变量和成员方法。例如: public class Parent { private int age…

    Java 2023年5月26日
    00
  • Java掩码的几种使用例举

    Java掩码的几种使用例举 在Java中,掩码的主要作用是用来过滤或者匹配不同的字节位。掩码是用位运算符来创建的。在Java中,我们可以使用按位与、或、异或等位运算符来创建掩码。 按位与掩码 按位与掩码是将每个位分别与操作数进行运算,返回新的结果。当操作数均为1的时候,该位的掩码返回1,否则返回0。在Java中,我们可以使用“&”符号来表示按位与掩码…

    Java 2023年5月29日
    00
  • spring框架_Applicationcontext功能

    Applicationcontext的功能拓展主要来自于不属于beanfactory的接口,主要包括四个接口 Messagesource :国际化 ResourcePatternResolver :获取资源 ApplicationEventPublisher:发布事件 EnvironmentCapable:获取环境变量 感觉发布事件这个功能有点像消息队列,发…

    Java 2023年5月9日
    00
  • 浅谈java web中常用对象对应的实例化接口

    我们来详细讲解一下“浅谈Java Web中常用对象对应的实例化接口”的完整攻略。 一、Java Web中常用对象的实例化接口 在Java Web开发中,常用的对象包括Servlet、JSP、HttpServletRequest、HttpServletResponse等。这些对象的实例化需要通过相应的接口来完成。 具体来说,常用对象的实例化接口如下: Serv…

    Java 2023年6月15日
    00
合作推广
合作推广
分享本页
返回顶部