首页 游戏 软件 资讯 排行榜 专题
首页
编程语言
c#如何动态编译代码_c#动态编译代码项目实例附完整源码

c#如何动态编译代码_c#动态编译代码项目实例附完整源码

热心网友
66
转载
2026-05-05

CSharpCodeProvider 在 .NET 6+ 中已彻底废弃,必须改用 Microsoft.CodeAnalysis(Roslyn)动态编译

c#如何动态编译代码_c#动态编译代码项目实例附完整源码

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

如果你仍在尝试使用 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 命名空间下的编译器类已被彻底删除。即便你在项目文件中添加 true 配置,或手动安装 System.CodeDom NuGet 包,CSharpCodeProvider 的构造函数也会抛出 NotSupportedException 异常。

  • 原生 CSharpCodeProvider 仅在目标框架为 net48netcoreapp3.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。即使存在沙箱环境,unsafestackalloc 或 P/Invoke 等代码仍可能突破限制。生产环境中,务必对源码的抽象语法树(AST)节点进行白名单校验。

调试环节同样需注意:编译错误信息存储在 Diagnostics 集合中,其行号对应原始字符串位置,而非文件行号。出错时,需自行按 \n 拆分源码字符串并标注行号——此细节常被忽略,却极易在部署初期引发问题。

来源:https://www.php.cn/faq/2333138.html
免责声明: 游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

相关攻略

怎样在SQL存储过程中调用C#编写的程序集_利用CLR集成技术实现
数据库
怎样在SQL存储过程中调用C#编写的程序集_利用CLR集成技术实现

在SQL Server存储过程中调用C 程序集:一份避坑指南 想在SQL Server的存储过程里直接调用C 代码?这个想法很自然,毕竟有些复杂计算或已有 NET逻辑,用T-SQL重写既麻烦又低效。SQL Server的CLR(公共语言运行时)集成功能,正是为此而生。但请注意,这并非简单的“混搭编程

热心网友
05.06
c#如何调用Python脚本_c#Python脚本的最佳实践与常见坑点
编程语言
c#如何调用Python脚本_c#Python脚本的最佳实践与常见坑点

C 调用Python脚本:最佳实践与常见坑点解析 使用 Process Start 调用 Python 脚本:最直接但需注意路径与环境 在大多数情况下,Process Start 是实现C 调用Python脚本最快捷的方案。它无需引入额外的NuGet包,也不强制要求Python解释器必须配置在系统环

热心网友
05.05
c#如何定义常量_c#定义常量的3种方式
编程语言
c#如何定义常量_c#定义常量的3种方式

C 常量定义:const、static readonly与静态类的实战指南 在C 编程实践中,常量的定义是基础但至关重要的环节。选择不当的常量声明方式,可能会为项目引入难以察觉的隐患。本文将深入解析C 中定义常量的三种核心方式:const、static readonly以及使用静态类进行封装,帮助你

热心网友
05.05
c#如何使用MEF框架_c#MEF框架的正确用法与注意事项
编程语言
c#如何使用MEF框架_c#MEF框架的正确用法与注意事项

CompositionContainer 初始化失败常因类型反射加载失败,主因是程序集版本 框架不匹配、DLL未显式加载或缺失部署依赖;Import为null则多因Catalog未包含对应Export、路径错误或契约不一致。 为什么 CompositionContainer 初始化失败常报“Unab

热心网友
05.05
C#怎么压缩并解压ZIP文件_C#如何管理压缩包【实战】
编程语言
C#怎么压缩并解压ZIP文件_C#如何管理压缩包【实战】

C 怎么压缩并解压ZIP文件_C 如何管理压缩包【实战】 说到在C 里处理ZIP文件,一个核心原则是:System IO Compression 是最稳妥的 ZIP 压缩方案。这意味着,你需要显式设置压缩级别为 CompressionLevel Optimal,使用正确的 ZipArchiveMod

热心网友
05.05

最新APP

宝宝过生日
宝宝过生日
应用辅助 04-07
台球世界
台球世界
体育竞技 04-07
解绳子
解绳子
休闲益智 04-07
骑兵冲突
骑兵冲突
棋牌策略 04-07
三国真龙传
三国真龙传
角色扮演 04-07

热门推荐

红米Note11 Pro更新系统需连WiFi吗?
电脑教程
红米Note11 Pro更新系统需连WiFi吗?

红米Note 11 Pro系统升级,为何坚持要求连接Wi-Fi? 当红米Note 11 Pro收到MIUI或澎湃OS的系统更新推送时,官方总会明确提示:整个过程请在Wi-Fi网络环境下完成。这项要求并非随意设定,而是基于清晰的技术与体验考量。一次完整的系统升级包,其大小通常在2GB至4GB之间。如果

热心网友
05.05
小米13ultra有nfc功能吗
电脑教程
小米13ultra有nfc功能吗

小米13 Ultra的NFC功能深度解析:它如何重新定义“全场景智能交互”? 在旗舰手机领域,NFC功能看似已成为标配,但体验却千差万别。小米13 Ultra所搭载的全功能NFC方案,在“全能”与“好用”两个维度上树立了新的标杆。它不仅无缝集成了公交卡模拟、门禁卡复制、数字车钥匙等核心生活服务,更全

热心网友
05.05
嵌入式消毒柜电源插座位置必须外露吗?
电脑教程
嵌入式消毒柜电源插座位置必须外露吗?

嵌入式消毒柜电源插座安装指南:隐蔽式布局提升安全与美观 在规划嵌入式消毒柜的安装方案时,电源插座的布局方式直接影响到最终的整体效果与安全性。正确的做法是避免插座外露,采用隐蔽式安装。根据国家《住宅厨房设计规范》及主流厨电品牌的安装标准,推荐将插座预留在消毒柜后方或侧方的墙体内部,安装高度宜控制在距地

热心网友
05.05
魔音耳机操作说明包含充电指示吗?
电脑教程
魔音耳机操作说明包含充电指示吗?

是的,魔音(Beats)耳机充电状态一目了然,指示灯明确显示 当你为Beats头戴式耳机充电时,如何判断它是否已经充满?答案就藏在机身自带的五段式LED电量指示灯里。在充电过程中,这排指示灯会持续闪烁,实时反馈充电进度。一旦所有五个指示灯全部转为稳定常亮、不再闪烁,即代表电池已完全充满。整个充电周期

热心网友
05.05
博朗剃须刀如何识别型号?
电脑教程
博朗剃须刀如何识别型号?

博朗剃须刀型号全解析:从编码规则到选购技巧的终极指南 面对博朗剃须刀复杂的字母数字组合感到困惑?实际上,其型号命名体系逻辑严谨,是用户选购的核心依据。简单来说,型号首位的数字(1、3、5、7、9)直接代表产品系列,数字越大,通常意味着技术越先进、功能越全面、定位越高端。例如,顶级的9系旗舰机型普遍搭

热心网友
05.05