即时编译器(JIT)的编译过程是什么?

1. 触发编译

JIT编译的触发基于热点代码检测,主要通过两种计数器:
方法调用计数器:统计方法被调用的次数(默认阈值:C1为1,500次,C2为10,000次)。
回边计数器:统计循环体的执行次数(如forwhile中的跳转指令)。

当计数器超过阈值时,JVM将代码加入编译队列,由后台编译线程处理。


2. 编译流程(以HotSpot JVM为例)

JIT的编译过程分为多个阶段,不同编译器(C1/C2/Graal)细节略有差异,但核心流程相似:

(1) 字节码解析与中间表示(IR)生成

输入:字节码(.class文件中的指令)。
输出:生成高级中间表示(HIR),如控制流图(CFG)。
关键操作
• 解析字节码,构建基本块(Basic Block)。
• 识别循环结构、异常处理块等。

(2) 方法内联(Method Inlining)

优化目标:减少方法调用开销。
规则
• 内联小方法(如getter/setter)。
• 避免内联巨型方法(通过-XX:MaxInlineSize=35控制阈值)。
示例

// 内联前
int getValue() { return value; }
void print() { System.out.println(getValue()); }// 内联后
void print() { System.out.println(value); }

(3) 逃逸分析(Escape Analysis)

目标:判断对象是否仅在方法内部使用。
优化结果
栈上分配:对象直接分配在栈帧,减少GC压力。
标量替换:将对象拆解为基本类型局部变量。
锁消除:若对象无竞争,移除同步操作。
示例

// 优化前
void foo() {Object obj = new Object();  // 对象可能被分配在堆synchronized(obj) { ... }   // 同步锁
}// 优化后(逃逸分析确认obj未逃逸)
void foo() {// 对象被标量替换或栈上分配,锁被消除
}

(4) 循环优化

循环展开(Loop Unrolling):减少循环条件判断次数。

// 优化前
for (int i = 0; i < 4; i++) { sum += i; }// 优化后
sum += 0; sum += 1; sum += 2; sum += 3;

循环向量化(SIMD):利用CPU单指令多数据(如AVX指令)并行计算数组。

(5) 公共子表达式消除(CSE)

移除冗余计算:复用相同表达式的计算结果。

// 优化前
int a = x * y + z;
int b = x * y + w;// 优化后
int tmp = x * y;
int a = tmp + z;
int b = tmp + w;

(6) 死代码消除(Dead Code Elimination)

移除不可达代码:如if (false) { ... }或未使用的变量。

// 优化前
int x = 1;
if (false) { x = 2; }  // 死代码
System.out.println(x);// 优化后
System.out.println(1);

(7) 本地代码生成

目标平台适配:根据CPU架构(x86/ARM)生成机器码。
寄存器分配:优化寄存器使用,减少内存访问。
指令选择:选择高效指令(如用LEA替代乘法)。


3. 分层编译(Tiered Compilation)

现代JVM(如HotSpot)采用分层策略逐步优化代码:

  1. 第0层:解释执行(快速启动)。
  2. 第1层(C1编译器):轻量优化(方法内联、简单循环展开)。
  3. 第2层(C2编译器):深度优化(逃逸分析、向量化)。
  4. Graal编译器(可选):替代C2,支持更激进优化(如JDK17的GraalVM)。

触发条件
• C1编译:方法调用约1,500次(-XX:CompileThreshold)。
• C2编译:方法调用约10,000次(-XX:Tier4CompileThreshold)。


4. 反优化(Deoptimization)

当假设条件不满足时,JVM会回退到解释执行,常见场景:
类加载变化:如动态加载新类导致原有优化失效。
多态调用:虚方法的目标方法改变(如接口实现类新增)。
断言失败:逃逸分析假设被打破(对象意外逃逸)。


5. 监控与调优

(1) 查看JIT编译日志

java -XX:+PrintCompilation -XX:+PrintInlining -XX:+UnlockDiagnosticVMOptions MyApp

输出示例

42   3    java.lang.String::hashCode (55 bytes)   COMPILE SKIPPED: already compiled
56   1    java.util.ArrayList::size (5 bytes)   INLINE (hot)

(2) 关键JVM参数

参数作用
-XX:+TieredCompilation启用分层编译(默认开启)
-XX:CompileThreshold=1500调整C1编译的调用阈值
-XX:MaxInlineSize=35控制方法内联的最大字节码大小
-XX:+PrintAssembly打印生成的机器码(需HSDIS插件)

6. 实际案例:优化斐波那契数列

原始代码(递归)

long fib(int n) {if (n <= 1) return n;return fib(n-1) + fib(n-2);  // 重复计算,性能差
}

JIT优化后

  1. 内联:递归调用被展开。
  2. 循环展开:转换为迭代形式。
  3. 最终机器码:近似以下优化版本:
    long fib(int n) {if (n <= 1) return n;long a = 0, b = 1;for (int i = 2; i <= n; i++) {long c = a + b;a = b;b = c;}return b;
    }
    

性能提升:从指数级时间复杂度(O(2^n))降至线性(O(n))。


7. 总结

触发条件:基于热点代码的调用或循环次数。
核心优化:方法内联、逃逸分析、循环优化、死代码消除等。
分层编译:逐步从解释执行过渡到深度优化(C1→C2)。
反优化:动态适应运行时变化,保证正确性。
调优建议:监控编译日志,合理配置阈值,避免冷启动影响性能测试。

理解JIT编译过程有助于编写对编译器友好的代码(如小方法、减少动态绑定),最大化运行时性能。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/75606.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java基础:集合List、Map、Set(超详细版)

集合体系概述 Collection常用方法 补充&#xff1a;addAll() Collection的遍历方式 迭代器 增强for&#xff08;空集合可以&#xff0c;null不可以&#xff09; lambda 集合对象存储对象原理 遍历方式的区别 List集合 特点、特有方法 遍历方式 &#xff08;同上&#xff09…

Elasticsearch 全面解析

Elasticsearch 全面解析 前言一、简介核心特性应用场景 二、核心原理与架构设计1. 倒排索引&#xff08;Inverted Index&#xff09;2. 分片与副本机制&#xff08;Sharding & Replication&#xff09;3. 节点角色与集群管理 三、核心特点1. 灵活的查询语言&#xff08;Que…

【2】k8s集群管理系列--包应用管理器之helm(Chart语法深入应用)

一、Chart模板&#xff1a;函数与管道 常用函数&#xff1a; • quote&#xff1a;将值转换为字符串&#xff0c;即加双引号 • default&#xff1a;设置默认值&#xff0c;如果获取的值为空则为默认值 • indent和nindent&#xff1a;缩进字符串 • toYaml&#xff1a;引用一…

JVM 字节码是如何存储信息的?

JVM 字节码是 Java 虚拟机 (JVM) 执行的指令集&#xff0c;它是一种与平台无关的二进制格式&#xff0c;在任何支持 JVM 的平台上都可运行的Java 程序。 字节码存储信息的方式&#xff0c;主要通过以下几个关键组成部分和机制来实现&#xff1a; 1. 指令 (Opcodes) 和 操作数 …

基于51单片机语音实时采集系统

基于51单片机语音实时采集 &#xff08;程序&#xff0b;原理图&#xff0b;PCB&#xff0b;设计报告&#xff09; 功能介绍 具体功能&#xff1a; 系统由STC89C52单片机ISD4004录音芯片LM386功放模块小喇叭LCD1602按键指示灯电源构成 1.可通过按键随时选择相应的录音进行播…

关于 Java 预先编译(AOT)技术的详细说明,涵盖 GraalVM 的配置、Spring Boot 3.x 的集成、使用示例及优缺点对比

以下是关于 Java 预先编译&#xff08;AOT&#xff09;技术的详细说明&#xff0c;涵盖 GraalVM 的配置、Spring Boot 3.x 的集成、使用示例及优缺点对比&#xff1a; 1. 预先编译&#xff08;AOT&#xff09;技术详解 1.1 核心概念 AOT&#xff08;Ahead-of-Time&#xff09…

【ROS2】行为树:BehaviorTree

1、简介 与状态机不同,行为树强调执行动作,而不是状态之间的转换。 行为树是可组合的。可以重复使用简单的行为来构建复杂的行为。 在游戏领域,行为树已经比较流行了。主要用于维护游戏角色的各种动作和状态。 ROS2的导航框架Navigation2中引入了行为树来组织机器人的工作流…

Centos7.9 升级内核,安装RTX5880驱动

系统镜像下载 https://vault.centos.org/7.9.2009/isos/x86_64/CentOS-7-x86_64-DVD-2009.iso 系统安装步骤省略 开始安装显卡驱动 远程登录查看内核 [root192 ~]# uname -a Linux 192.168.119.166 3.10.0-1160.el7.x86_64 #1 SMP Mon Oct 19 16:18:59 UTC 2020 x86_64 x8…

多层感知机与全连接神经网络关系解析

感知机&#xff08;Perceptron&#xff09;、多层感知机&#xff08;MLP&#xff0c;Multilayer Perceptron&#xff09;和全连接神经网络&#xff08;FCNN&#xff0c;Fully Connected Neural Network&#xff09;是神经网络发展过程中密切相关的概念&#xff0c;但它们有明确…

解析医疗器械三大文档:DHF、DMR与DHR

医疗器械的 DHF、DMR 和 DHR 是质量管理体系&#xff08;QMS&#xff09;中的核心文件&#xff0c;贯穿产品全生命周期&#xff0c; 确保医疗器械的安全性、有效性和合规性。 一、三大文件的定义与法规依据 缩写全称法规依据&#xff08;以 FDA 为例&#xff09;核心目的DHF…

netty启用websocket的压缩机制

netty启用websocket的压缩机制 package com.aerotop.connector.websocket.base;import io.netty.channel.ChannelInitializer; import io.netty.channel.ChannelPipeline; import io.netty.channel.socket.SocketChannel; import io.netty.handler.codec.compression.JZlibDec…

可能存在特殊情况,比如控制台显示有延迟、缓冲问题等影响了显示顺序。

从控制台输出看&#xff0c;正常逻辑应是先执行 System.out.println(" 未处理异常演示 "); 输出对应文本&#xff0c;再因 arr 为 null 访问 length 触发 NullPointerException 输出异常信息。可能存在特殊情况&#xff0c;比如控制台显示有延迟、缓冲问题等影响…

第一节:React 基础篇-React虚拟DOM原理及Diff算法优化策略

必考点&#xff1a;虚拟DOM树对比&#xff08;同级比较、Key的作用、组件类型判断&#xff09; 延伸&#xff1a;React 18中并发更新对Diff算法的影响 React虚拟DOM原理及Diff算法优化策略 虚拟DOM核心原理 概念&#xff1a; • 虚拟DOM&#xff08;Virtual DOM&#xff09;…

Java spring mybatis面试题(200道),八股文

Java面试题 通过网盘分享的文件&#xff1a;面试题等2个文件 链接: https://pan.baidu.com/s/1Xw0PzkfAmL8uesYBvrW2-A?pwdpebt 提取码: pebt mybatis相关 1、 什么是Mybatis&#xff1f; … 2 2、 Mybaits 的优点&#xff1a; … 2 3、 MyBatis 框架的缺点&#xff1a; ……

windows使用Python调用7-Zip【按大小分组】压缩文件夹中所有文件

使用Python调用7-Zip【按大小分组】压缩文件夹中所有文件 问题描述&#xff1a;方法前提条件任务完整代码示例代码如何工作&#xff1f; 问题描述&#xff1a; 我现在想把文件夹下的所有内容上传到网盘&#xff0c;但是这个文件夹下的素材内容很多&#xff0c;使用分卷压缩的话…

《Python星球日记》第26天:Matplotlib 可视化

名人说&#xff1a;路漫漫其修远兮&#xff0c;吾将上下而求索。—— 屈原《离骚》 创作者&#xff1a;Code_流苏(CSDN)&#xff08;一个喜欢古诗词和编程的Coder&#x1f60a;&#xff09; 订阅专栏&#xff1a;《Python星球日记》 目录 一、Matplotlib 简介1. 什么是 Matplo…

第1章 对大型语言模型的介绍

人类正处在一个关键转折点。自2012年起&#xff0c;基于深度神经网络的人工智能系统研发进入快速通道&#xff0c;将这一技术推向了新高度&#xff1a;至2019年底&#xff0c;首个能够撰写与人类文章真假难辨的软件系统问世&#xff0c;这个名为GPT-2&#xff08;生成型预训练变…

dcsdsds

我将为您在页面顶部添加欢迎内容&#xff0c;同时保持整体风格的一致性。以下是修改后的代码&#xff0c;主要修改了模板部分和对应的样式&#xff1a; vue 复制 <template><div class"main-wrapper"><!-- 新增欢迎部分 --><div class"…

学习MySQL的第八天

海到无边天作岸 山登绝顶我为峰 一、数据库的创建、修改与删除 1.1 引言 在经过前面七天对于MySQL基本知识的学习之后&#xff0c;现在我们从基本的语句命令开始进入综合性的语句的编写来实现特定的需求&#xff0c;从这里开始需要我们有一个宏观的思想&…

Linux-内核驱动-中断-key

DEV_NAME&#xff1a;设备名称。 wg&#xff1a;等待队列头&#xff0c;用于同步。 condition&#xff1a;条件变量&#xff0c;用于等待和唤醒。 中断处理函数 irq_handler&#xff1a;处理中断请求&#xff0c;设置条件变量并唤醒等待队列中的进程。 文件操作函数 open…