上一片介绍了Primitive层,Attribute Model可以认为是对Primitive的上层实现。主要包括如下内容:

1. 一系列的Attribute来定义Import和Export

常用的有如下Attribute:Export,InheritedExport,Import,ImportConstructor,ImportMany,ExportMetadata

Export:如果需要将某个class定义为可供其他组件Import的组件,只需将Export attribute应用于该class即可。这相当于申明了一个ExportDefinition.

例如:

[Export]
class Person
{
public string Name { get; set; } public string Age { get; set; }
}

这里将Person类导出,依赖Person的组件可以通过Import Attribute来定义依赖。

ExportAttribute有两个属性,ContractName和Type,其实都是用来标识Export的,用来在Import的时候指定依赖目标。在没有显示设置情况下,默认是用Type来标识。

InheritedExport是将Export定义在基类上,所有派生类自动应用基类的Export

Import: 其他组件如果需要依赖声明为Export的组件,则使用Import Attribute来指定。

例如:

class Company
{
  [Import]
  private Person _chairMan
}

这里Company类定义了自身对Person组件的依赖。ImportAttribute也有ContractName和Type属性,用来定义所依赖组件的标识符。

ImportConstructor:通过该Attributes来定义容器创建组件是使用的构造函数,函数的参数自动定义为Import,会通过容器来注入。

ImportMany:导入所有满足条件的组件实例。参考如下代码:

public interface ILog
{ } [Export(typeof(ILog))]
public class FileLog : ILog
{ } [Export(typeof(ILog))]
public class DBLog : ILog
{ } public class TaskExecutor
{
[ImportMany(typeof(ILog))]//声明注入所有提供ILog接口的组件实例
public IEnumerable<ILog> _loggers;
} class Program
{ static void Main(string[] args)
{ var container = new CompositionContainer(new ApplicationCatalog()); var executor = new TaskExecutor();
container.ComposeParts(executor);// FileLog 和 DBLog都会注入_logger数组 Console.ReadKey();
}
}

ExportMetadata:有时候ImportMany的时候我们不希望对满足导出接口的所有组件实例进行进一步筛选.这个时候可以通过ExportMetadata来定义Export的组件具有哪些特性,然后在依赖组件中对导入的组件根据Metadata进行筛选。依然用上面的例子,可以这样来进行筛选。

[Export(typeof(ILog))]
[ExportMetadata("Type", "db")]
public class DBLog : ILog
{ }

  //需要定义一个接口来声明Export所支持的metadata
public interface ILogMetadata
{
string Type { get; }
} public class TaskExecutor
{
[ImportMany(typeof(ILog))]
public IEnumerable<Lazy<ILog, ILogMetadata>> _loggers;//这里将变量类型声明为Lazy<ILog,ILogMetadata>,这样在依赖注入的时候不会创建组件的实例 public IEnumerable<ILog> AvailableLogger
{
get
{
var result = new List<ILog>();
foreach (var logger in _loggers)
{
            //对metadata进行筛选
if (logger.Metadata.Type == "db")
{
result.Add(logger.Value);
}
} return result;
}
}
}

CreationPolicy:

在将组件通过ExportAttribute导出的时候,可以通过PartCreationPolicyAttributes来指定该组件的实例创建法则.有如下几种选项:

Any //将选择权交给Container来决策,该选项是默认选项,而Container层一般是选择Shared

NonShared //每次填充依赖的时候都创建一个新的组件实例来满足Import

Shared //以单例的形式创建组件实例

同时在定义Import的时候也能指定所需求的组件实例的形式。可以通过设置Import的RequiredCreationPolicy属性来指定。这样就会出现Import的CreationPolicy和Export的CreationPolicy不一致的问题。两者组合最终得到的结果如下:

|| Part.Any     ||     Part.Shared     ||     Part.NonShared || 
| Import.Any              | Shared        |        Shared           |     Non Shared         | 
| Import.Shared         | Shared        |        Shared           |     No Match            | 
| Import.NonShared  | Non Shared |        No Match      |     Non Shared         |

2. 一系列发现和生成ComposablePartDefinition的ComposablePartCatalog

PartCatalog是按一定规则来发现Composable Part,并生成对应的ComposablePartDefinition。MEF提供了如下几种Catalog:

System.ComponentModel.Composition.Primitives.ComposablePartCatalog
    System.ComponentModel.Composition.Hosting.AggregateCatalog //可以组合几种Catalog
    System.ComponentModel.Composition.Hosting.ApplicationCatalog //在应用程序目录中的Dll和Exe中发现组件
    System.ComponentModel.Composition.Hosting.AssemblyCatalog //在指定的Assembly中发现组件
    System.ComponentModel.Composition.Hosting.CompositionScopeDefinition //这个还没用过
    System.ComponentModel.Composition.Hosting.DirectoryCatalog //在指定目录中的dll和exe中发现组件
    System.ComponentModel.Composition.Hosting.FilteredCatalog //基于指定的Catalog来应用一个Filter,产生新的Catalog
    System.ComponentModel.Composition.Hosting.TypeCatalog //基于一组类型来发现组件

当将Catalog注册到Container中之后,Catalog会自动去发现Composable Part,并车安生对应的Definition.

3. AttributedModelService

除了定义上述的规则,然后让Container去自动发现组件,生成PartDefinition,和创建Part外。AttributedModelService类提供了若干方法可以手动的去创建PartDefinition,创建Part,添加Part,注入依赖。

[Export]
public class Person
{
public string Name { get; set; } public string Age { get; set; }
} [Export]
public class Company
{
[Import]
public Person ChairMan { get; set; }
} class Program
{ static void Main(string[] args)
{
var container = new CompositionContainer(new ApplicationCatalog()); var company = new Company();
var person = new Person(); var personPart = AttributedModelServices.CreatePart(person);
var companyPart = AttributedModelServices.CreatePart(company); var batch = new CompositionBatch();
batch.AddPart(personPart);
batch.AddPart(companyPart); container.Compose(batch); Console.ReadKey();
}

上述代码就是通过手动创建Part,然后注册到Container,然后通过调用Compose方法并注入依赖。

AttributedModelService也提供了扩展方法来直接面向组件实例编程,而不用了解到Primitive层。

通过调用扩展方法ComposeParts(CompositionContainer, Object[]),可以直接传入通过Attribute定义好Import/Export的组件实例对象。组件的依赖会自动注入,并将Export注册到Container.

MEF学习总结(3)---Attribute Model Programing的更多相关文章

  1. MEF学习总结(1)---总体架构

    用了很久的MEF框架来做依赖注入,最近想把它的原理和机构总结一下,主要包括如下几个方面: 1. 总体架构 2. .Net Composition Primitive 3. Attribute Mode ...

  2. C#可扩展编程之MEF学习笔记(五):MEF高级进阶

    好久没有写博客了,今天抽空继续写MEF系列的文章.有园友提出这种系列的文章要做个目录,看起来方便,所以就抽空做了一个,放到每篇文章的最后. 前面四篇讲了MEF的基础知识,学完了前四篇,MEF中比较常用 ...

  3. C#可扩展编程之MEF学习笔记(四):见证奇迹的时刻

    前面三篇讲了MEF的基础和基本到导入导出方法,下面就是见证MEF真正魅力所在的时刻.如果没有看过前面的文章,请到我的博客首页查看. 前面我们都是在一个项目中写了一个类来测试的,但实际开发中,我们往往要 ...

  4. C#可扩展编程之MEF学习笔记(三):导出类的方法和属性

    前面说完了导入和导出的几种方法,如果大家细心的话会注意到前面我们导出的都是类,那么方法和属性能不能导出呢???答案是肯定的,下面就来说下MEF是如何导出方法和属性的. 还是前面的代码,第二篇中已经提供 ...

  5. C#可扩展编程之MEF学习笔记(二):MEF的导出(Export)和导入(Import)

    上一篇学习完了MEF的基础知识,编写了一个简单的DEMO,接下来接着上篇的内容继续学习,如果没有看过上一篇的内容, 请阅读:http://www.cnblogs.com/yunfeifei/p/392 ...

  6. C#可扩展编程之MEF学习笔记(一):MEF简介及简单的Demo

    在文章开始之前,首先简单介绍一下什么是MEF,MEF,全称Managed Extensibility Framework(托管可扩展框架).单从名字我们不难发现:MEF是专门致力于解决扩展性问题的框架 ...

  7. C#可扩展编程之MEF学习

    MEF系列文章: C#可扩展编程之MEF学习笔记(一):MEF简介及简单的Demo C#可扩展编程之MEF学习笔记(二):MEF的导出(Export)和导入(Import) C#可扩展编程之MEF学习 ...

  8. [转]学习C#:Attribute与Property

    一.什么是Attribute 先看下面的三段代码: 1.自定义Attribute类:VersionAttribute [AttributeUsage(AttributeTargets.Class)] ...

  9. AngularJs学习笔记--Understanding the Model Component

    原版地址:http://docs.angularjs.org/guide/dev_guide.mvc.understanding_model 在angular文档讨论的上下文中,术语“model”可以 ...

随机推荐

  1. ubuntu12.04+openni+nit+SensorKinect环境搭建

    一.安装openni 1.下载openni                  OpenNI-Bin-Dev-Linux-x64-v1.5.4.0.tar.bz2 2.cd ~; mkdir kinec ...

  2. docker 快速搭建 WordPress

    安装Docker 环境:阿里云服务器 镜像:CentOs 7.4 64 https://docs.docker.com/install/linux/docker-ce/centos/ 安装WordPr ...

  3. PHP for循环的写法和示例

    For循环是最近的循环语句之一,无论哪种语言,都有这个循环语句,也是我们工作中常用的循环方法. 语法规则: for (expr1; expr2; expr3){ 要执行的代码 } expr1:表示循环 ...

  4. zset类型以及其操作

    sorted set类型 sorted sets类型以及其操作zset是set的一格升级版本,它在set的基础上增加了一格顺序属性,这一属性在添加元素的同时可以指定,每次指定后,zset会自动重新按照 ...

  5. 备注下Windows可能会用到的运行命令

    因为有几个命令不常用忘记了,所以备注下Windows可能会用到的运行命令: 1.cleanmgr:打开磁盘清理工具2.compmgmt.msc:计算机管理3.charmap:启动字符映射表4.calc ...

  6. python 中 property 属性的讲解及应用

    Python中property属性的功能是:property属性内部进行一系列的逻辑计算,最终将计算结果返回 property属性的有两种方式: 1. 装饰器 即:在方法上应用装饰器 2. 类属性 即 ...

  7. APUE学习笔记——5缓冲Buffering、流、文件对象

    缓冲的几个基本概念     缓冲的作用:减少系统read和write的次数. 全缓冲         系统标准I/O缓冲区被写满时才进行真正的I/O操作.         磁盘文件一般使用全缓冲   ...

  8. 20165202 学习基础和c语言基础调查

    你有什么技能比大多人(超过90%以上)更好? 我对自行车运动的兴趣始于初中时期,不敢说比大多数人更好,但在业余爱好者中相对来说还不错. 针对这个技能的获取你有什么成功的经验? 接触自行车运动几年里,我 ...

  9. Linux系统在启动过程中内核文件丢失的解决方法

    在/boot目录下有两个重要的文件,分别是: vmlinuz-3.10.0-123.el7.x86_64         内核文件 initamfs-3.10.0-123.el7.x86_64.img ...

  10. phpMyAdmin 尝试连接到 MySQL 服务器,但服务器拒绝连接

    phpMyAdmin 尝试连接到 MySQL 服务器,但服务器拒绝连接--解决方法     phpMyAdmin 尝试连接到 MySQL 服务器,但服务器拒绝连接.您应该检查配置文件中的主机.用户名和 ...