您的位置:首页 >c#如何动态编译代码_c#动态编译代码项目实例附完整源码
发布于2026-05-02 阅读(0)
扫一扫,手机访问

如果你还在尝试使用 CSharpCodeProvider 来动态编译代码,那么有一个坏消息:在 .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()
新的解决方案核心是使用 CSharpCompilation 配合 Emit 方法,将输出写入 MemoryStream,最后通过 AssemblyLoadContext.Load 加载。流程比旧方式稍显繁琐,但带来的好处是显而易见的:控制力更强、完全跨平台,并且规避了旧方案中的反射漏洞风险。
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" 或 "Test"。但在 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) })。动态编译绝非玩具,一旦计划在生产环境中使用,就必须设定清晰的边界:
CancellationToken。一个常见的做法是用 Task.Run(...).Wait(timeout) 将其包裹起来,超时则取消任务并释放相关的 AssemblyLoadContext。LoadFromStream 都可能注册一个新的上下文。如果不调用 AssemblyLoadContext.Unload,这些程序集将永久驻留内存。但 .NET 6+ 的默认上下文是不可卸载的,因此建议自定义一个派生类,并重写设置 IsCollectible = true。ParseText。即使有沙箱环境,unsafe、stackalloc 或 P/Invoke 等代码仍有可能逃逸限制。在生产环境中,务必对源码的抽象语法树(AST)节点进行白名单校验。其实,最让人头疼的往往是调试环节:编译错误信息都藏在 Diagnostics 集合里,其中的行号对应的是原始字符串中的位置,而非文件行号。一旦出错,你就得自己动手,按 \n 拆分源码字符串并标注行号——这个细节很少被提及,但上线第一天就很可能撞上。
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
正版软件
正版软件
正版软件
正版软件
正版软件
1
2
3
7
9