首页 游戏 软件 资讯 排行榜 专题
首页
编程语言
c#如何使用MEF框架_c#MEF框架的正确用法与注意事项

c#如何使用MEF框架_c#MEF框架的正确用法与注意事项

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

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

c#如何使用MEF框架_c#MEF框架的正确用法与注意事项

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

为什么 CompositionContainer 初始化失败常报“Unable to load one or more of the requested types”

当你在 C# 中使用 MEF 框架遇到此错误时,通常不是框架本身的问题,而是类型反射加载环节出现了故障。容器在运行时无法定位到你所定义的类型。这主要源于以下几种典型情况:程序集版本不匹配、目标框架不一致(例如插件基于 .NET 5 编译,而宿主程序运行在 .NET 6 环境),或者包含目标类型的 DLL 未被正确加载到当前应用程序域中。

  • 确保程序集显式可用:所有参与组合的部件,其所在的程序集都必须通过 Assembly.LoadFrom 等方法显式加载,或确保它们已存在于 AppDomain.CurrentDomain.GetAssemblies() 的列表中。
  • 警惕未部署的依赖:如果你的 [Export] 类型引用了某些仅在开发环境中存在的 NuGet 包(如特定的调试工具包),而这些依赖未随主程序一同部署,运行时将无法找到它们。
  • 善用异常诊断:一个高效的排查技巧是使用 try/catch 捕获 ReflectionTypeLoadException,并检查其 LoaderExceptions 属性。它能精确指出具体是哪个类型加载失败,是定位问题的关键工具。

Import 属性为 null?检查 ComposablePartCatalog 是否包含对应 Export

一个常见的误区是认为 MEF 会自动扫描整个应用程序来寻找导出部件。实际上,MEF 只会从你显式提供ComposablePartCatalog(目录)中进行查找。如果你使用了 DirectoryCatalog,但插件 DLL 存放路径错误、文件扩展名不符、甚至文件被安全软件锁定,都可能导致 Import 属性返回 null。

  • 确认目录内容:直接检查 catalog.Parts.Count()。如果结果为 0,基本可以确定是路径错误或程序集加载失败。
  • 注意文件类型DirectoryCatalog 默认仅识别 .dll 文件,对于 .exe 或无扩展名的文件,它不会进行加载。
  • 别忘了主程序集:如果导出类定义在主执行程序集内,务必使用 new AssemblyCatalog(Assembly.GetExecutingAssembly()) 将其添加到目录中。
  • 契约必须严格匹配:导入属性上必须标注 [Import],且其类型签名必须与 [Export] 的定义完全一致,包括命名空间。当使用接口导出时,如果指定了契约名称(contract name),那么导入时也必须使用相同的契约名称才能成功匹配。

ExportFactory 替代直接 Import 实现延迟创建与生命周期控制

直接使用 [Import] 存在一个局限:对象会在组合容器构建完成时立即实例化,这使你无法控制创建时机,也难以管理其生命周期和资源释放。而 ExportFactory 正是为解决此问题而设计的。它提供了 CreateExport() 方法,允许你在需要时才创建实例,每次调用都会生成一个新的实例,并支持通过 Dispose() 进行显式释放。

  • 导出端配置:在导出类上,通常需要这样标注:[Export(typeof(IMyService))][PartCreationPolicy(CreationPolicy.NonShared)]
  • 导入端声明:不再直接导入服务,而是导入其工厂:[Import] public ExportFactory ServiceFactory { get; set; }
  • 使用模式:实际使用时,遵循标准模式:using var export = ServiceFactory.CreateExport(); var service = export.Value;
  • 关键提醒:如果导出类实现了 IDisposable,并且创建策略是 NonShared(非共享),那么务必在不再需要时手动调用 export.Dispose(),否则可能导致实例泄漏和资源未释放。

.NET Core/.NET 5+ 中为什么 System.ComponentModel.Composition 不推荐继续用

对于现代 .NET 项目,需要明确:原生的 MEF(即 System.ComponentModel.Composition 命名空间下的 API)在 .NET Core 及更高版本中,更多是作为兼容层存在,功能上存在诸多限制。例如,它不支持 DirectoryCatalog 的热重载、处理跨平台路径时可能行为不一致、反射机制也与传统的 .NET Framework 版本有所差异。更重要的是,微软已将其标记为“legacy”(遗留技术)。因此,新项目应当考虑迁移。

  • 主流替代方案:首选是转向 Microsoft.Extensions.DependencyInjection 结合自定义的插件加载逻辑。
  • 社区 MEF2:另一个选择是使用社区维护的 System.Composition(常被称为 MEF2)。它是一个轻量级替代品,API 设计相似但并不兼容旧版 MEF,需要替换命名空间,并使用 ContainerConfiguration 来构建容器。
  • MEF2 的优势:它提供了更精细的控制,例如通过 AssemblyPartDiscovery 可以更精确地筛选类型,避免意外加载测试类或内部工具类。
  • 注意差异:MEF2 没有内置的 Lazy 自动包装机制。如果需要延迟加载,需手动用 Lazy 包裹 ExportFactory 来实现。

总而言之,MEF 的许多问题都源于对“自动化”的过度依赖。它看似自动完成了所有依赖注入,但实际上,每一处连接是否畅通、接口是否完全匹配,都需要开发者仔细确认。尤其是在跨程序集的复杂场景下,类型加载、生命周期管理和契约匹配这三个核心环节,任何一个出现偏差,都可能导致 Import 属性静默地变为 null,而不会抛出明确的错误信息。这正是使用 MEF 进行插件式开发时需要高度警惕的地方。

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

相关攻略

c#如何绘制图形_c#绘制图形的正确用法与注意事项
编程语言
c#如何绘制图形_c#绘制图形的正确用法与注意事项

C 绘图避坑指南:从Graphics来源到DPI适配的实战要点 在C 中进行图形绘制,一个看似简单的DrawRectangle背后,往往藏着好几个“坑”。Graphics对象不能直接new,否则要么直接报错,要么静默失败——所有绘图操作都必须基于合法的来源。这可以说是入门绘图的第一条铁律。 Grap

热心网友
05.05
VSCode怎么搭建Unity 3D的C#脚本编写环境并解决找不到引用的问题
编程语言
VSCode怎么搭建Unity 3D的C#脚本编写环境并解决找不到引用的问题

VSCode怎么搭建Unity 3D的C 脚本编写环境并解决找不到引用的问题 在Unity开发中,用VSCode写C 脚本时遇到“找不到引用”的红色波浪线,这事儿确实挺让人头疼的。别急,这通常不是代码逻辑问题,而是开发环境之间的“沟通”出了岔子。下面咱们就来逐一拆解最常见的几个原因和对应的解决方案。

热心网友
05.04
C#如何使用Record类型_C#不可变数据模型特性解析【极简】
编程语言
C#如何使用Record类型_C#不可变数据模型特性解析【极简】

C Record类型:不可变数据容器的正确打开方式 先明确一个核心认知:C 中的Record类型,本质上是一个“省心”的不可变数据容器。它不是什么更高级的class,而是编译器帮你自动生成值相等性、ToString、GetHashCode以及with表达式的语法糖。用对了,它能帮你省掉80%的数据

热心网友
05.03
C#如何获取硬件信息_C# WMI读取CPU与硬盘序列号【进阶】
编程语言
C#如何获取硬件信息_C# WMI读取CPU与硬盘序列号【进阶】

WMI无法稳定读取现代CPU与NVMe硬盘序列号?问题不在代码,而在硬件与系统本身 一个常见的开发误区是:用WMI读取CPU和硬盘序列号,结果发现拿不到、拿不准或者拿到一堆乱码。问题往往不在于你的代码写错了,而是系统或固件层面,压根就没把这个“身份证号”暴露给你。 为什么 Win32_Process

热心网友
05.02
C#怎么防止UI线程假死_C#耗时操作放入后台线程更新UI【核心】
编程语言
C#怎么防止UI线程假死_C#耗时操作放入后台线程更新UI【核心】

C 怎么防止UI线程假死_C 耗时操作放入后台线程更新UI【核心】 耗时操作必须离开 UI 线程,否则假死不可避免 —— 这不是优化建议,而是 WinForms WPF 的运行铁律。 为什么直接在 Button_Click 里调用 Thread Sleep 就卡死? 道理其实很简单:UI 线程身兼数

热心网友
05.01

最新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