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 拆分源码字符串并标注行号——此细节常被忽略,却极易在部署初期引发问题。
游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。
同类文章
Go语言中Struct Tag详解:XML解析必备的字段标签机制
Go语言Struct Tag深度解析:XML数据绑定与字段映射的核心机制 Struct Tag是Go语言为结构体字段附加元数据的核心语法,广泛应用于XML、JSON等数据序列化场景。它通过反引号包裹的键值对进行声明,本质上是指导编码器与解码器如何精确映射结构体字段与外部数据格式。缺少它,Go程序将无
c#如何调用Python脚本_c#Python脚本的最佳实践与常见坑点
C 调用Python脚本:最佳实践与常见坑点解析 使用 Process Start 调用 Python 脚本:最直接但需注意路径与环境 在大多数情况下,Process Start 是实现C 调用Python脚本最快捷的方案。它无需引入额外的NuGet包,也不强制要求Python解释器必须配置在系统环
c#如何定义常量_c#定义常量的3种方式
C 常量定义:const、static readonly与静态类的实战指南 在C 编程实践中,常量的定义是基础但至关重要的环节。选择不当的常量声明方式,可能会为项目引入难以察觉的隐患。本文将深入解析C 中定义常量的三种核心方式:const、static readonly以及使用静态类进行封装,帮助你
c#如何使用MEF框架_c#MEF框架的正确用法与注意事项
CompositionContainer 初始化失败常因类型反射加载失败,主因是程序集版本 框架不匹配、DLL未显式加载或缺失部署依赖;Import为null则多因Catalog未包含对应Export、路径错误或契约不一致。 为什么 CompositionContainer 初始化失败常报“Unab
C#怎么压缩并解压ZIP文件_C#如何管理压缩包【实战】
C 怎么压缩并解压ZIP文件_C 如何管理压缩包【实战】 说到在C 里处理ZIP文件,一个核心原则是:System IO Compression 是最稳妥的 ZIP 压缩方案。这意味着,你需要显式设置压缩级别为 CompressionLevel Optimal,使用正确的 ZipArchiveMod
- 日榜
- 周榜
- 月榜
1
2
3
4
5
6
7
8
9
10
1
2
3
4
5
6
7
8
9
10
相关攻略
2015-03-10 11:25
2015-03-10 11:05
2021-08-04 13:30
2015-03-10 11:22
2015-03-10 12:39
2022-05-16 18:57
2025-05-23 13:43
2025-05-23 14:01
热门教程
- 游戏攻略
- 安卓教程
- 苹果教程
- 电脑教程
热门话题

