.NET Core 和 .NET Framework 中的 MEF2
MEF,Managed Extensibility Framework,现在已经发布了三个版本了,它们是 MEF 和 MEF2。
等等!3 去哪儿了?本文将教大家完成基于 MEF2 的开发。
MEF 和 MEF2
其实微软发布了四个版本的 MEF:
- 随着 .NET Framework 4.0 发布,微软称之为 MEF
- 随着 .NET Framework 4.5 发布,微软让它更好用了,微软称之为 MEF2,但因为接口兼容,也直接称之为 MEF
- .NET 开发团队觉得 MEF 第一代性能太差,于是通过 NuGet 为移动设备发布了可移植类库,是个轻量级版本,只移植了 .NET Framework 中 MEF2 里 2 的部分;随后 .NET Core 中也加入了 MEF2,也是 .NET Framework 中 MEF2 里 2 的部分
- Visual Studio 开发团队觉得 .NET Framework 里的 MEF2 性能太差,NuGet 版的 MEF2 功能太少,于是自己又写了一个,微软称之为 VS-MEF
对于第一代的 MEF,我们这里就完全不说了,性能又差功能又少,没有利用价值。
对于 .NET Framework 4.5 里引入的 MEF2,性能上没能改进多少,倒是使用起来功能更多。详细资料和使用方法请参考微软官方的文档:
- Managed Extensibility Framework (MEF) - Microsoft Docs
- Attributed Programming Model Overview (MEF) - Microsoft Docs
而本文主要说的 MEF2 是微软后来以 NuGet 包形式发布的 MEF2;适用于 .NET Framework 4.5 及以上、.NET Core 和各种 .NET 移动平台。它的接口相比于 .NET Framework 中原生带的已经变了,中文和英文的参考资料很少,几乎都是参考微软官方发布的文档才能使用。所以本文将为大家提供其中文的使用方法指导。
至于性能提升程度,我没有进行定量测试,所以直接从 IoC Container Benchmark - Performance comparison - www.palmmedia.de 一文中搬运了性能测试结果,如下:
安装 MEF2
.NET Framework 中自带的 MEF 在程序集 System.ComponentModel.Composition.dll 中,命名空间为 System.ComponentModel.Composition
。MEF2 随 NuGet 包发布,其 NuGet 包名是 Microsoft.Composition,命名空间为 System.Composition
。
所以,在需要使用 MEF2 的项目中安装以上 NuGet 包即可完成安装。
使用 MEF2 开发
MEF 完全使用特性来管理容器中的依赖,微软称之为 Attributed Programming Model,并辅以广告——不需要配置文件的依赖注入容器。所以,使用特性来标记依赖关系就成了 MEF 的招牌依赖管理方式。
使用方法我将分为两个部分来讲,最容易的是业务代码,给开发团队中所有成员使用的代码。比较难的是框架代码,给开发团队中写框架的那一部分成员。
业务代码
业务代码的写法其实取决于框架开发者怎么去定义框架。但是,为了方便大家理解,在这一节我将只说 MEF2 最原生的使用方法。框架那一节我才会说明如何自定义业务代码的写法。
最原生的使用方法其实只有两个——[Import]
和 [Export]
,其它都是变种!具体说来,标记了 Export
的类将导出给其它类使用;标记了 Import
的属性/字段/方法参数等将接收来自 Export
的那些类/属性/字段的实例。
Import/Export
在类型上标记 [Export]
可以让容器发现这个类型。[Export]
允许带两个参数,一个契约名称,一个契约类型。在 [Import]
的时候,相同的契约名称会被注入;与属性或字段的类型相同的契约类型会被注入。
IEnumerable/Lazy
如果属性或字段是集合类型,可以使用 [ImportMany] 来注入集合(如果 Export
有多个)。
如果属性或字段是 Lazy<T>
类型,那么并不会立即注入,而是在访问到 Lazy<T>.Value
时才获取到实例(如果此时的创建过程由容器处理,那么第一次访问 Value
时才会创建)。
框架代码
框架代码也分为两个部分:一个部分是初始化,初始化后可以创建一个依赖注入容器;另一个部分是管理依赖,将使用之前初始化好的依赖注入容器进行管理。
初始化的最简代码如下:
var compositionHost = new ContainerConfiguration().CreateContainer();
那么,得到的 compositionHost 变量将是用来管理依赖的容器,你可以将它储存在字段中用于随后管理依赖。
但是,只是这么初始化将得不到任何对象。所以,我们需要额外添加配置代码,以便将一些程序集中的对象添加到容器中:
var compositionHost = new ContainerConfiguration().WithAssemblies(new []
{typeof(A).Assembly,typeof(B).Assembly,typeof(C).Assembly,typeof(D).Assembly,
}).CreateContainer();
这样,A/B/C/D 这四个类分别所在的程序集中,直接或间接加了 [Export]
特性的类都将被此依赖容器管理。
MEF2 之所以为 2,因为它除了能通过 [Export]
特性导出,还能直接在框架中发现而不必由业务开发者手动指定。这在第三方代码或者不希望被 MEF 侵入的代码中非常有用。例如,我们将所有已有的 ViewModel 导出:
// 使用 ConventionBuilder 自动导出所有的 ViewModel。
var convention = new ConventionBuilder();// 将所有继承自 ViewModelBase 的类导出,并共享一个实例(即注入到多个属性中的都是同一个实例)。
convention.ForTypesDerivedFrom<ViewModelBase>().Export().Shared();// 使用这些配置创建依赖注入容器。
var compositionHost = new ContainerConfiguration().WithAssemblies(new []
{typeof(A).Assembly,typeof(B).Assembly,typeof(C).Assembly,typeof(D).Assembly,
}).WithDefaultConventions(convention).CreateContainer();;
注意,以上代码中的 .Shared()
目的是让导出的 ViewModel
共享实例(同一个类型的实例只有一个)。
只初始化是不行的,还需要将这些依赖注入到目标实例中才行。使用 SatisfyImports
可以将传入的对象中的所有依赖注入进去。
compositionHost.SatisfyImports(targetObject);
在框架设计中,对于不同模块中的类型,框架需要决定使用哪一个容器来注入,或者是否注入。所以上面这个代码会发生在使用 MEF2 框架中需要注入的任何一个部分。
参考资料
- MEF in .NET 4.5 - CodeProject
- Managed Extensibility Framework(MEF) 2 框架新特性介绍 - PetterLiu - 博客园
- Is MEF or MEF2 baked into the .NET Framework? - Stack Overflow
- vs-mef/why.md at master · Microsoft/vs-mef
- mef/Home.md at master · MicrosoftArchive/mef
.NET Core 和 .NET Framework 中的 MEF2相关推荐
- 登录注册的小项目对比.Net Core与 .Net Framework的一些区别
一.需求: 1.功能只有登录.注册. 二.架构: 1.分别为 UserSys.IServices:主要有实体和对实体的配置,还有对实体的操作接口 UserSys.Services :主要是对自ISer ...
- .NET Core中延迟单例另一种写法【.NET Core和.NET Framework的beforefieldinit差异】
1.BeforeFieldInit是什么 前段时间在反编译代码时无意间看到在类中有一个BeforeFieldInit特性,处于好奇的心态查了查这个特性,发现这是一个关于字段初始化时间的特性[提前初始化 ...
- 如何在 Entity Framework 中计算 时间差 ?
咨询区 ison 我的项目中有一个需求,需要使用 Entity Framework 实现 日期差 的计算逻辑,参考如下代码: var now = DateTime.UtcNow;db.Items.Or ...
- .NET Core 和 .NET Framework 启动可执行文件的差别
在 Windows 下,使用 .NET Framework 构建出来的应用,可以只有一个可执行文件,在可执行文件里面包含了 IL 代码.使用 .NET Core 构建出来的应用,将会包含一个 Exe ...
- ASP.NET Core 开发-Entity Framework (EF) Core 1.0 Database First
ASP.NET Core 开发-Entity Framework Core 1.0 Database First,ASP.NET Core 1.0 EF Core操作数据库. Entity Frame ...
- .NET Core 和 .NET Framework 之间的关系
引用一段描述:Understanding the relationship between .NET Core and the .NET Framework. .NET Core and the .N ...
- Android Framework中的Application Framework层介绍
Android的四层架构相比大家都很清楚,老生常谈的说一下分别为: Linux2.6内核层,核心库层,应用框架层,应用层.我今天重点介绍一下应用框架层Framework. Framework层为我们开 ...
- 使用ASP.NET Core和Entity Framework Core实现Angular 7 SPA CRUD
目录 Angular 7 Angular Core变化 Angular CLI更改 升级到Angular 7 添加新项目 创建新项目 使用Angular的前端设计和实现 添加模型和配置文件 添加Boo ...
- asp向不同的用户发送信息_.Net Core 和 .Net Framework的不同
起因 近期因为公司业务的关系,开始使用.net生态环境,因此对.Net Core和 .Net Framework进行了一些研究. 历史 .Net Framework 历史要比 .Net Core 久一 ...
最新文章
- vue代码上传服务器后背景图片404解决方法
- python需要php吗-C、C+、Java、PHP、Python分别用来开发什么?
- ubuntu复制文件到另一个文件夹_简单介绍一下电脑中的文件或文件夹的复制、移动及删除的操作方式...
- redhat怎样修改语言_硕士博士个人陈述(PS)辅导及修改服务带你极速前进!
- word文档一级计算机,计算机一级-word讲义
- 2017-2018-1 20155301 《信息安全系统设计基础》第十三周学习总结
- SLAM_视觉SLAM中的一种单目稠密建图方法
- 给大家安利一个买电脑好去处(内有福利)
- 1)⑤爬取搜狗旅游部分新闻
- SDN免费快速获得积分和直接获取下载码的几个办法,亲测有效区
- Java快捷键的设置与使用
- 摩西十诫 摩西简介 世界宗教图谱
- 查询硬盘序列号(serialNumber),系统安装日期等
- U盘被写保护无法格式化的解决方法
- 10分钟快速学Handlebars
- 【uniapp】悬浮球(floatball)全局组件——全局消息提醒功能
- 著名电子竞技游戏分析
- 解决SQLSever配置管理器不见了
- ZJS-415/DC220V中间继电器
- ps钢笔工具/路径/钢笔抠图/字体工具
热门文章
- 计算机联锁控制系统的软件应具备信号操作功能,车站信号计算机联锁控制系统—软件.ppt...
- sqrt函数实现(神奇的算法)
- 是否允许应用获取设备信息_手机权限获取弹窗不断,隐私安全如何确保,这些权限需谨慎...
- onReachBottom无法触发也不报错
- 陆奇:冬天过后一定是春天,首先要活下来
- Selenium(二)_控件定位及简单操作
- 电脑打不开u盘,总显示“本次操作由于这台计算机的限制而被取消,请与您的系统管理员联系”
- 计算机另存找不到桌面,“文件保存到桌面但是找不到”的解决方案
- 材料库存天数的计算方法
- 中国科学院大学计算机软件与理论量子研究生,2019中国科学院大学硕士研究生入学考试《量子力学》考试大纲...