c#如何动态编译代码_c#动态编译代码项目实例附完整源码
CSharpCodeProvider 在 .NET 6+ 中已彻底废弃,必须改用 Microsoft.CodeAnalysis(Roslyn)动态编译

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈
如果你仍在尝试使用 CSharpCodeProvider 来实现 C# 代码的动态编译,那么请注意:在 .NET 6 及更高版本中,此方法已完全失效。运行时将直接抛出 Type or namespace name 'CSharpCodeProvider' could not be found 错误。该技术已成为历史,仅适用于 .NET Framework 及早期的 .NET Core 3.1 项目。
为什么 CSharpCodeProvider 编译失败或根本找不到
这是开发者升级项目时首先遇到的障碍。自 .NET 5 起,微软默认移除了对传统 CodeDOM 的支持,而在 .NET 6+ 中,Microsoft.CSharp 命名空间下的编译器类已被彻底删除。即便你在项目文件中添加 配置,或手动安装 System.CodeDom NuGet 包,CSharpCodeProvider 的构造函数也会抛出 NotSupportedException 异常。
- 原生
CSharpCodeProvider仅在目标框架为net48或netcoreapp3.1的项目中可用。 - 目标框架为
net5.0及以上的项目,必须迁移至Microsoft.CodeAnalysis(即 Roslyn 编译器)。 - 若项目目标为
net6.0,代码中仍包含new CSharpCodeProvider(),编译虽可通过,但运行时必然崩溃。
使用 Roslyn 替代方案:将字符串代码编译至内存程序集
新的解决方案核心是使用 CSharpCompilation 配合 Emit 方法,将输出写入 MemoryStreamAssemblyLoadContext.Load 加载。流程虽比旧方案稍复杂,但优势显著:提供更强的控制力、完全跨平台,并有效规避了旧方案中的反射安全漏洞。
- 首先,需通过 NuGet 安装包:
Microsoft.CodeAnalysis.CSharp(建议使用 v4.0+ 版本)。 - 必须显式添加所有依赖的程序集引用。例如,
System.Console位于System.Runtime.dll中,不能仅模糊引用"System.dll"。 - 常用快捷方式是使用
MetadataReference.CreateFromFile(Assembly.GetExecutingAssembly().Location)引用当前程序集。但需注意:若执行程序集为单文件发布(即设置了PublishTrimmed=true),.Location属性将返回空字符串。此时需采用备用方案,如使用typeof(object).Assembly.Location。 - 以下是关键代码示例:
var syntaxTree = CSharpSyntaxTree.ParseText(code); var references = new List{ MetadataReference.CreateFromFile(typeof(object).Assembly.Location), MetadataReference.CreateFromFile(typeof(Console).Assembly.Location), // 添加其他必要引用... }; var compilation = CSharpCompilation.Create( assemblyName: $"dyn_{Guid.NewGuid():N}", syntaxTrees: new[] { syntaxTree }, references: references, options: new CSharpCompilationOptions(OutputKind.DynamicallyLinkedLibrary)); using var ms = new MemoryStream(); var result = compilation.Emit(ms); if (!result.Success) { var errors = result.Diagnostics.Where(d => d.IsWarningAsError || d.Severity == DiagnosticSeverity.Error); throw new InvalidOperationException(string.Join("\n", errors.Select(e => e.ToString()))); } ms.Seek(0, SeekOrigin.Begin); var assembly = AssemblyLoadContext.Default.LoadFromStream(ms);
动态调用方法时类型名与命名空间的常见错误
旧方案常依赖硬编码的类型名,如 "DynamicClass"。但在 Roslyn 编译后,类型名完全由源代码中的 class 声明决定。若代码定义为 namespace MyNS { public class Worker { ... } },则必须使用 assembly.GetType("MyNS.Worker") 获取类型,遗漏命名空间将返回 null。
- 实用建议:在构造源码字符串时,强制指定命名空间。例如:
@"using System; namespace Dyn { public class Entry { public static void Run() { Console.WriteLine(""ok""); } } }"。 - 调用静态方法使用
type.GetMethod("Run").Invoke(null, null);调用实例方法需先通过Activator.CreateInstance(type)创建对象。 - 若源代码包含泛型或异步方法,使用
GetMethod时必须传入精确的方法签名。例如:GetMethod("RunAsync", new[] { typeof(CancellationToken) })。
安全与性能的关键约束
动态编译技术若用于生产环境,必须设定明确的边界以确保稳定与安全:
- 超时控制:Roslyn 编译过程本身不支持
CancellationToken。常见做法是用Task.Run(...).Wait(timeout)包裹编译任务,超时则取消并释放相关的AssemblyLoadContext。 - 内存泄漏:每次调用
LoadFromStream都可能注册新的上下文。若不调用AssemblyLoadContext.Unload,程序集将永久驻留内存。由于 .NET 6+ 的默认上下文不可卸载,建议自定义派生类并设置IsCollectible = true。 - 源码安全校验:严禁将未经处理的用户输入直接传入
ParseText。即使存在沙箱环境,unsafe、stackalloc或 P/Invoke 等代码仍可能突破限制。生产环境中,务必对源码的抽象语法树(AST)节点进行白名单校验。
调试环节同样需注意:编译错误信息存储在 Diagnostics 集合中,其行号对应原始字符串位置,而非文件行号。出错时,需自行按 \n 拆分源码字符串并标注行号——此细节常被忽略,却极易在部署初期引发问题。
相关攻略
在SQL Server存储过程中调用C 程序集:一份避坑指南 想在SQL Server的存储过程里直接调用C 代码?这个想法很自然,毕竟有些复杂计算或已有 NET逻辑,用T-SQL重写既麻烦又低效。SQL Server的CLR(公共语言运行时)集成功能,正是为此而生。但请注意,这并非简单的“混搭编程
C 调用Python脚本:最佳实践与常见坑点解析 使用 Process Start 调用 Python 脚本:最直接但需注意路径与环境 在大多数情况下,Process Start 是实现C 调用Python脚本最快捷的方案。它无需引入额外的NuGet包,也不强制要求Python解释器必须配置在系统环
C 常量定义:const、static readonly与静态类的实战指南 在C 编程实践中,常量的定义是基础但至关重要的环节。选择不当的常量声明方式,可能会为项目引入难以察觉的隐患。本文将深入解析C 中定义常量的三种核心方式:const、static readonly以及使用静态类进行封装,帮助你
CompositionContainer 初始化失败常因类型反射加载失败,主因是程序集版本 框架不匹配、DLL未显式加载或缺失部署依赖;Import为null则多因Catalog未包含对应Export、路径错误或契约不一致。 为什么 CompositionContainer 初始化失败常报“Unab
C 怎么压缩并解压ZIP文件_C 如何管理压缩包【实战】 说到在C 里处理ZIP文件,一个核心原则是:System IO Compression 是最稳妥的 ZIP 压缩方案。这意味着,你需要显式设置压缩级别为 CompressionLevel Optimal,使用正确的 ZipArchiveMod
热门专题
热门推荐
红米Note 11 Pro系统升级,为何坚持要求连接Wi-Fi? 当红米Note 11 Pro收到MIUI或澎湃OS的系统更新推送时,官方总会明确提示:整个过程请在Wi-Fi网络环境下完成。这项要求并非随意设定,而是基于清晰的技术与体验考量。一次完整的系统升级包,其大小通常在2GB至4GB之间。如果
小米13 Ultra的NFC功能深度解析:它如何重新定义“全场景智能交互”? 在旗舰手机领域,NFC功能看似已成为标配,但体验却千差万别。小米13 Ultra所搭载的全功能NFC方案,在“全能”与“好用”两个维度上树立了新的标杆。它不仅无缝集成了公交卡模拟、门禁卡复制、数字车钥匙等核心生活服务,更全
嵌入式消毒柜电源插座安装指南:隐蔽式布局提升安全与美观 在规划嵌入式消毒柜的安装方案时,电源插座的布局方式直接影响到最终的整体效果与安全性。正确的做法是避免插座外露,采用隐蔽式安装。根据国家《住宅厨房设计规范》及主流厨电品牌的安装标准,推荐将插座预留在消毒柜后方或侧方的墙体内部,安装高度宜控制在距地
是的,魔音(Beats)耳机充电状态一目了然,指示灯明确显示 当你为Beats头戴式耳机充电时,如何判断它是否已经充满?答案就藏在机身自带的五段式LED电量指示灯里。在充电过程中,这排指示灯会持续闪烁,实时反馈充电进度。一旦所有五个指示灯全部转为稳定常亮、不再闪烁,即代表电池已完全充满。整个充电周期
博朗剃须刀型号全解析:从编码规则到选购技巧的终极指南 面对博朗剃须刀复杂的字母数字组合感到困惑?实际上,其型号命名体系逻辑严谨,是用户选购的核心依据。简单来说,型号首位的数字(1、3、5、7、9)直接代表产品系列,数字越大,通常意味着技术越先进、功能越全面、定位越高端。例如,顶级的9系旗舰机型普遍搭





