0

0

C# JIT编译过程 C#即时编译器是如何工作的

煙雲

煙雲

发布时间:2026-02-16 09:49:02

|

473人浏览过

|

来源于php中文网

原创

jit 编译是按需延迟编译:方法首次调用时才将 il 编译为本地代码,支持线程安全、分层编译(tier 0/tier 1)、运行时优化,并可通过环境变量或工具(如 complus_jitdisasm、dotnet-dump)调试分析。

c# jit编译过程 c#即时编译器是如何工作的

CLR 加载 IL 后不会立刻编译,而是延迟到方法首次调用时才触发 JIT

这是 JIT 的核心机制:不是整个程序启动时一并编译,而是一个方法一个方法地“按需编译”。mscorlib.dll 或你自己的 MyApp.dll 被加载进进程后,其中的 IL(中间语言)只是被验证和元数据解析,真正生成本地机器码发生在第一次执行某个方法时。

比如调用 Console.WriteLine(),CLR 会检查该方法是否已被编译;若没有,就触发 JIT 编译器对这个方法的 IL 块做一次翻译,生成 x64 或 ARM64 指令,并缓存到内存中。后续再调用它就直接跳转到已编译的本地代码,跳过编译开销。

  • JIT 编译是线程安全的,多个线程同时首次调用同一方法,只会有一个线程真正执行编译,其余等待
  • 编译结果只在当前 AppDomain(.NET Framework)或 AssemblyLoadContext(.NET Core/5+)内有效,跨上下文不共享
  • 方法内联、循环展开等优化由 JIT 在运行时决定,取决于当前 CPU 架构和运行时配置(如 DOTNET_JIT_OPTIMIZATIONTier

不同 .NET 版本的 JIT 引擎差异明显,特别是 Tiered Compilation 的引入

.NET Core 3.0 起默认启用分层编译(Tiered Compilation),它让 JIT 不再“一次编译、终身使用”,而是分两级:

  • Tier 0:快速生成低优化度代码(几乎无内联、无循环优化),目标是降低启动延迟;方法被调用几次后触发重编译
  • Tier 1:完整优化编译(启用所有 JIT 优化策略),替换掉 Tier 0 的代码;触发条件通常是方法被热路径执行(如循环体、高频 API)

你可以用环境变量临时关闭它来观察行为差异:DOTNET_TieredCompilation=0。关闭后所有方法都走 Tier 1 流程,启动慢但稳态性能略高;开启后冷启动快,且长期运行下热点方法仍能获得最佳性能。

注意:.NET Native(UWP)和 AOT 编译(如 dotnet publish -r win-x64 --aot)完全绕过 JIT,IL 在构建时就被编译为原生代码,此时 System.Runtime.CompilerServices.JitHelpers 等类型不可用。

JIT 编译失败时常见错误是 MethodImplOptions.AggressiveOptimization 或泛型约束引发的验证失败

不是所有 IL 都能被 JIT 接受。典型失败场景包括:

OpenMP并行程序设计 WORD版
OpenMP并行程序设计 WORD版

本文档主要讲述的是OpenMP并行程序设计;OpenMP是一个编译器指令和库函数的集合,主要是为共享式存储计算机上的并行程序设计使用的。目前支持OpenMP的语言主要有Fortran,C/C++。 OpenMP在并行执行程序时,采用的是fork/join式并行模式,共享存储式并行程序就是使用fork/join式并行的。希望本文档会给有需要的朋友带来帮助;感兴趣的朋友可以过来看看

下载
  • AggressiveOptimization 方法里用了不支持的 IL 指令(如某些未验证的指针操作),JIT 会拒绝编译并抛出 NotSupportedException
  • 泛型方法含复杂约束(如 where T : new(), IComparable<t>, IDisposable</t>)且实例化时类型不满足全部约束,可能在 JIT 时才发现——错误信息常是 InvalidProgramExceptionVerificationException(.NET Framework)
  • 调试模式下启用了 DebuggableAttribute.DebuggingModes.DisableOptimizations,可能导致某些优化路径失效,间接影响 JIT 行为

这类问题往往只在 Release 模式、特定 CPU 架构(如 ARM64)、或 Tier 1 编译阶段暴露,开发阶段不易复现。建议用 dotnet trace 或 PerfView 抓取 JIT 日志(启用 Microsoft-Windows-DotNETRuntime/JIT/MethodJittedVerbose 事件)定位具体哪个方法卡住。

想看 JIT 实际干了什么?用 COMPLUS_JitDisasmdotnet-dump 查看生成的汇编

最直接的方式是让 JIT 输出汇编指令。例如,在 Windows 上设置环境变量后运行程序:

set COMPLUS_JitDisasm=MyNamespace.MyClass::MyMethod
dotnet run

控制台就会打印出该方法经 JIT 编译后的 x64 汇编(含寄存器分配、指令选择细节)。注意它只对首次调用生效,且仅限当前进程。

更通用的做法是用 dotnet-dump 分析运行中进程:

  • dotnet-dump collect -p <pid></pid> 保存内存快照
  • dotnet-dump analyze <dumpfile></dumpfile>,然后输入 clrstack -a 找到目标线程和方法地址
  • 最后用 dumpil <methodaddr></methodaddr> 看 IL,dumpmt -md <methodaddr></methodaddr> 查 JIT 状态,u <nativeaddr></nativeaddr> 反汇编本地代码

这些输出里能看到 JIT 如何把 for (int i = 0; i 优化成无边界检查的循环,或如何把 <code>if (obj is string s) 编译成单条 test + je 指令——但前提是没被内联进调用方。

真正难调试的永远不是“JIT 没工作”,而是“它工作得太聪明”:比如把一个看似复杂的判断折叠成常量,或因内联导致堆栈无法映射回原始 C# 行号。这时候得关掉内联([MethodImpl(MethodImplOptions.NoInlining)])再对比。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

750

2023.08.02

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1553

2023.10.24

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

814

2023.08.22

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

199

2023.11.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

750

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

569

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

234

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

209

2025.08.29

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

283

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 9.5万人学习

C 教程
C 教程

共75课时 | 4.7万人学习

C++教程
C++教程

共115课时 | 17.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号