前言

在前面的介绍中我们已经知道:导入和导出的匹配成功需要ContractType,ContractName,Metadata都匹配,这里我们还要介绍一个新的东西:创建策略(creation policy )。有时候我们在容器中的实例在每一个导入之间共享,即单例;有时候我们需要让每一个导入都拥有一个各自的实例,这在MEF中就是有创建策略决定的。

生命周期(Life Cycle),每一个MEF的部件在容器中都有自己的生命周期,何时创建,何时释放等。本文就主要介绍一下MEF中得创建策略和生命周期

创建策略

创建策略,其实就是组合容器决定如何创建部件。在组合容器组合部件时,如果导入和导出匹配成功,则组合容器会将导入成员的值设置成为导出的实例。因此,导出部件的创建策略决定了部件来源于何处:是现有实例还是新实例。

MEF的创建策略有:Shared(共享)和NonShared(非共享)。

使用Shared创建策略的部件将在每一个导入部件中共享实例。仅当容器中没有该部件的实例时才会创建新实例。使用共享策略创建的部件可以是提供服务的部件,以及较占用内存的部件。他们的内部状态应该尽可能少得受外界影响。

使用NonShared策略的部件在匹配每一个导入时都会有新的实例。这些部件的内部状态都是相互独立的,当某些部件需要保持特定的状态时可以使用这种策略。

MEF关于创建策略提供的特性有3种:Shared,NonShared,Any(请注意Any和Shared的区别,创建策略只有两种)。导入和导出的默认值都是Any。当导入导出都是用默认创建策略,或者都是用默认,MEF将默认创建策略为Shared。创建策略的匹配与否也决定了导出部件和导入能否成功匹配。以下情况可视为匹配成功:

1. 标记为Any的导出部件与Shared和NonShared的导入均能成功匹配;

2. 创建策略为Shared或NonShared的导出以及标记为Any的导入匹配成功;

3. 创建策略为Shared的导出只能与创建策略为Shared和标记为Any匹配成功;

4. 创建策略为NonShared的导出只能为Shared和标记为Any的导入匹配成功。

注意:

MSDN是这么表述的:
导入和导出的默认值均为 Any。 
指定 Shared 或 NonShared 的导出将仅与指定相同值或指定 Any 的导入匹配。 
同样,指定 Shared 或 NonShared 的导入将仅与指定相同值或指定 Any 的导出匹配。
如果导入和导出均指定 Any,或者未指定创建策略那么默认为 Any,则创建策略将默认为“共享“。

创建策略特性的用法:

//导入部件

[Import(RequiredCreationPolicy = CreationPolicy.Shared)]
//导出部件
[PartCreationPolicy(CreationPolicy.NonShared)]

请看一个简单的例子:

//定义导出部件,创建策略为非共享
[Export]
[PartCreationPolicy(CreationPolicy.NonShared)]
public class FileLog
{
public void WriteLog()
{
Console.Writeline("This is FileLog!");
}
} //定义导出部件,创建策略为非共享
[Export]
[PartCreationPolicy(CreationPolicy.Shared)]
public class DBLog
{
public void WriteLog()
{
Console.Writeline("This is DBLog!");
}
} public class MyClass
{
//成功匹配
[Import(RequiredCreationPolicy = CreationPolicy.NonShared)]
public FileLog MyFileLog1; //匹配失败
[Import(RequiredCreationPolicy = CreationPolicy.Shared)]
public FileLog MyFileLog2; //匹配成功
[Import]
public DBLog MyDBLog; }

注:到目前为止我们可以知道影响导出部件同导入部件成功匹配的主要因素大概有:ContractType(协定类型),ContractName(协定名称),Metadata(元数据),CetationPolicy(创建策略)。

生命周期

在MEF中,生命周期是比较复杂的,在MSDN中也只用一句话带过:

由于部件承载于组合容器中,因此其生命周期可能比普通对象更复杂。 部件可实现两个重要的生命周期相关接口:IDisposable 和 IPartImportsSatisfiedNotification。

容器本身实现了IDisposable,在容器Dispose方法被调用时,容器会对容器中得每一个部件调用Dispose方法。一般情况下,部件只有在容器释放时才会释放资源。联想到上面介绍的创建策略,有人不禁要问如果创建策略为非共享的部件过多,很占用很多资源,不过不用担心:容器提供了 ReleaseExport 方法。此方法可以释放导出部件,并对部件占用的资源进行释放。

如果部件实现了接口IPartImportsSatisfiedNotification ,当组合已完成并且部件的导入可供使用时,组合窗口将对部件调用接口中得方法OnImportsSatisfied。

IPartImportsSatisfiedNotification 包含一个名为 OnImportsSatisfied 的方法。 当组合已完成并且部件的导入可供使用时,组合窗口将对实现接口的任何部件调用此方法。 部件是组合引擎创建,用于满足其他部件的导入。 在设置好部件的导入之前,您无法执行任何依赖于部件构造函数中的导入值或对这些值进行操作的初始化,除非已通过使用 ImportingConstructor 特性将这些指定为必备。 此方法通常为首选方法,但在某些情况下,构造函数注入可能不可用。 在这些情况下,可以在 OnImportsSatisfied 中执行初始化,并且部件应实现 IPartImportsSatisfiedNotification。

灵活运用部件的生命周期在具体使用MEF中会有很大得帮助,如:我们可能将WCF,EF做成组件,然后使用MEF整合这些组件,那么在组合成功后,我们可能需要读取WCF配置发布服务,或者读取DB.config建立数据库连接等。这时在OnImportsSatisfied 中完成这些操作可能效果更好。

http://www.cnblogs.com/pszw/archive/2011/12/14/2286895.html

创建策略(Creation Policy )和生命周期(Life Cycle)的更多相关文章

  1. @Scope注解设置创建bean的方式和生命周期

    1.1.1            Scope注解创建bean的方式和生命周期 作用 Scope设置对象在spring容器(IOC容器)中的生命周期,也可以理解为对象在spring容器中的创建方式. 取 ...

  2. iOS开发之控制器创建与加载(生命周期)

    1.如何创建一个控制器 控制器常见的创建方式有以下几种: (1)通过storyboard创建 (2)直接创建 MJViewController *mj = [[MJViewController all ...

  3. Maven(二)Maven项目的创建(命令、myeclipse)及生命周期

    上一篇给大家介绍了Maven的概念和仓库的一些信息,接下来给大家分享一下使用命令和MyEclipse创建Maven项目 一.使用命令管理Maven项目 1.1.创建Maven java项目 1)创建一 ...

  4. Vue项目的创建、路由、及生命周期钩子

    目录 一.Vue项目搭建 1.环境搭建 2.项目的创建 3.pycharm配置并启动vue项目 4.vue项目目录结构分析 5.Vue根据配置重新构建依赖 二.Vue项目创建时发生了什么 三.项目初始 ...

  5. 启动独立的tomcat服务器,没有自动创建ServletContext,对Context生命周期的监听失败

    1.可能web.xml文件里对ContextListener没有进行配置 2.web.xml文件有关对ContextListener的配置,出现了错误的单词拼写问题 比如 <listener&g ...

  6. web forms page和control的生命周期life cycle交互,以及page生命周期中每个event中需要做什么事情

    只有 page_load和page_init这些可以autoeventwireup RenderControl只提供override public override void RenderContro ...

  7. 这么简单的ES索引生命周期管理,不了解一下吗~

    对于日志或指标(metric)类时序性强的ES索引,因为数据量大,并且写入和查询大多都是近期时间内的数据.我们可以采用hot-warm-cold架构将索引数据切分成hot/warm/cold的索引.h ...

  8. Elasticsearch索引生命周期管理方案

    一.前言 在 Elasticsearch 的日常中,有很多如存储 系统日志.行为数据等方面的应用场景,这些场景的特点是数据量非常大,并且随着时间的增长 索引 的数量也会持续增长,然而这些场景基本上只有 ...

  9. Logstash & 索引生命周期管理(ILM)

    Grok语法 Grok是通过模式匹配的方式来识别日志中的数据,可以把Grok插件简单理解为升级版本的正则表达式.它拥有更多的模式,默认,Logstash拥有120个模式.如果这些模式不满足我们解析日志 ...

随机推荐

  1. 31C3 CTF web关writeup

    0x00 背景 31c3 CTF 还是很人性化的,比赛结束了之后还可以玩.看题解做出了当时不会做的题目,写了一个writeup. 英文的题解可以看这:https://github.com/ctfs/w ...

  2. Ceph实战入门之安部署篇

    最近Ceph官方发布了luminous长久支持版,新版本增加了很多有意思的功能,但是入门还是先从部署安装开始. 环境说明 在Win10下安装VMware® Workstation 12 Pro软件,用 ...

  3. hive不分区增量更新

    insert overwrite table ods.zeg_so select *,case when zsm.id is not null then cast(current_timestamp ...

  4. SpringBoot 出现内置的Tomcat没有启动的情况

    Tomcat未启动,也未报错 pom.xml文件中增加 <dependency> <groupId>org.springframework.cloud</groupId& ...

  5. CI/CD----jenkins+gitlab+django(内网)

    1.py第三方包获取 ./pip3 ./pip3 -i "http://pypi.douban.com/simple/" --trusted-host pypi.douban.co ...

  6. A*算法实现(图形化表示)——C++描述

    概要 A*算法是一种启发式寻路算法,BFS是一种盲目的无目标的搜索算法,相比于BFS,A*算法根据适应度构建优先队列,根据适应度值可以很好的向目标点移动,具体详情,请看搜索相关文档,我在只是实现了在无 ...

  7. Bilibli文章无法复制文字

    在文章的正文部分右键,选择检查 删除箭头所指的内容unable-reprint

  8. 关于在react里面es6构建类的时候,一些开始的问题

    一般来说我们写react代码,个人习惯 这个里面没有constructor和super的,这样写也没啥问题.因为他会默认加上 但是有的时候有人会加上这两个 可以不写constructor,一旦写了co ...

  9. Python&R&Matlab:批量生成变量

    在编写程序时,有时我们需要命名相当多的变量,比如x0.x1.x2.....xn,用手一个个打出来是相当麻烦的.那么这时我们就需要批量生成变量了. 解决这个问题的关键在于,'xn'是自动构造出来的字符串 ...

  10. shiro框架学习-6-Shiro内置的Filter过滤器及数据加解密

    1.  shiro的核心过滤器定义在枚举类DefaultFilter 中,一共有11个 ,配置哪个路径对应哪个拦截器进行处理 // // Source code recreated from a .c ...