随笔-254  评论-3297  文章-48  trackbacks-74

MEF程序设计指南九:重组(Recomposition)MEF部件

  通过前篇八篇程序设计指南的学习,相信大家都知道应用程序部件的组合都是在初始化的时候进行装配的。如果当应用程序已经初始化完成了,此时又有新的部件被导入且进行装配,按照目前的实现方式就无法实现了,我们需要一种可以进行动态装配、动态组合以及可以动态的进行新的部件被装配组合的通知的功能,于此MEF所提供的重组(Recomposition)部件特性就可以派上用场了。

 

  这里还是使用上一篇中应用的示例作为功能演示场景,一个图书接口其下有3个不同的具体实现类被定义在主程序中,详细如下代码段: 

namespace MEFTraining.Recomposition
{
    [Export(
typeof(IBookService))]
    
public class MEFBookService : IBookService
    {
        
public string GetBookName()
        {
            
return "《MEF程序设计指南》";
        }
    }

    [Export(
typeof(IBookService))]
    
public class ASPNETBookService : IBookService
    {
        
public string GetBookName()
        {
            
return "《ASP.NET项目案例》";
        }
    }

    [Export(
typeof(IBookService))]
    
public class SLBookService : IBookService
    {
        
public string GetBookName()
        {
            
return "《Silverlight高级编程》";
        }
    }
}

 

  那么我们可以在需要使用到他们地方进行一个多项导入就可以导入所有的图书实现类,详细如下代码。其中AllowRecomposition=true参数就表示运行在有新的部件被装配成功后进行部件集的重组。

[ImportMany(AllowRecomposition = true)]
public IBookService[] Services { getset; }

 

  此时运行程序可以发现,Services的长度为3,表示成功的将上面三种图书的实现成功的装配了。接下来我们在一个新的Silverlight应用里面定义另外一个实现(OracleBookService),然后将此应用程序进行动态的加载并装配,这里MEF提供了DeploymentCatalog类,就是专门用来动态加载装配外部Silverlight应用程序包的,只需要提供应用程序包所在的路径就可以自动的完成下载和动态装配,使用方式可参考下面代码段,更详细请查看《MEF程序设计指南七:使用目录(Catalog)动态装载xap与目录筛选(Filtered Catalog)》一文中有详细介绍。

var dc = new DeploymentCatalog("MEFTraining.Recomposition.Parts.xap");
dc.DownloadAsync();
dc.DownloadCompleted 
+= (rs, re) =>
{ };

 

  下面代码段为定义的第四个图书实现部件,接下来将应用于重组功能演示示例。

namespace MEFTraining.Recomposition.Parts
{
    [Export(
typeof(IBookService))]
    
public partial class OracleBookService : UserControl,IBookService
    {
        
public OracleBookService()
        {
            InitializeComponent();
        }

        
public string GetBookName()
        {
            
return "《Oracle DBA手册》";
        }
    }
}

 

  使用DeploymentCatalog去加载第四个一个部件所在的应用程序包,最终通过调试可以发现Services已经进行了重组,已经从原来的3个部件变为了4个部件。

 this.DService.AddXap("MEFTraining.Recomposition.Parts.xap"null);

         

 

  在进行动态下载装配的过程中,主程序或许需要准确的得到一个部件被装配成功的通知,IPartImportsSatisfiedNotification接口就是一个当有新的部件进行装配成功后的一个通知接口,可以准确的监听到MEF容器的组合,一旦有新的插件部件进行导入装载到MEF容器中,此接口就会自动的得到通知。其内部就一个接口方法,详细如下代码块:

public void OnImportsSatisfied()
{
            
}

 

  下面为完整的主程序代码,本文完整示例代码请下载代码附件查看。

public partial class MainPage : UserControl, IPartImportsSatisfiedNotification
{
    [Import]
    
public IDeploymentService DService { getset; }

    [ImportMany(AllowRecomposition 
= true)]
    
public IBookService[] Services { getset; }

    
public MainPage()
    {
        InitializeComponent();
        CompositionInitializer.SatisfyImports(
this);
    }

    
private void button1_Click(object sender, RoutedEventArgs e)
    {
        
this.DService.AddXap("MEFTraining.Recomposition.Parts.xap"null);
    }

    
public void OnImportsSatisfied()
    {
            
    }
}

 

  参考资料:Recomposition , DeploymentCatalog

  MEF官方网站:http://mef.codeplex.com/

  推荐指南:MEF程序设计指南一:在应用程序中宿主MEF

         MEF程序设计指南二:Silverlight中使用CompositionInitializer宿主MEF

         MEF程序设计指南三:MEF中组合部件(Composable Parts)与契约(Contracts)的基本应用

       MEF程序设计指南四:使用MEF声明导出(Exports)与导入(Imports)

            MEF程序设计指南五:迟延(Lazy)加载导出部件(Export Part)与元数据(Metadata)

        MEF程序设计指南六:MEF中的目录服务(DeploymentCatalog)

                   MEF程序设计指南七:使用目录(Catalog)动态装载xap与目录筛选(Filtered Catalog)

        MEF程序设计指南八:部件生命周期(Parts Lifetime)托管

 

相关说明

 本文属学习笔记文章,愿与有志者共同学习交流。欢迎转载,但请在明显地位标记本文的原文连接。  

作      者:Beniao

文章出处:http://beniao.cnblogs.com/  或  http://www.cnblogs.com/

 

 

posted on 2010-07-31 00:47 Bēniaǒ 阅读(2687) 评论(7) 编辑 收藏

评论:
#1楼 2010-08-03 08:46 | mufeng100      
hi,请教一个问题:
一般地,类A中的字段导入类B(B中可能导入其他类),会得到B的一个实例对象。
如果想在A中获得B的一个数组,怎么做呢?注:不是一个抽象类的多个实现的数组,而是一个类的数组。

 回复 引用 查看   
#2楼[楼主] 2010-08-08 12:20 | Bēniaǒ      
@mufeng100
不明白你的意思。

 回复 引用 查看   
#3楼 2011-04-02 15:33 | sdfstasdfadfafd[未注册用户]
楼主提供源码下载呀
 回复 引用   
#4楼[楼主] 2011-04-06 16:35 | Bēniaǒ      
@sdfstasdfadfafd
引用sdfstasdfadfafd:楼主提供源码下载呀

文中有详细的示例代码。

 回复 引用 查看   
#5楼 2011-12-09 17:33 | FerventDesert      
楼主想咨询一个问题,
我的工程中,组件是有层次结构的,有些既是导入也是导出部件,我现在的做法是所有需要导入的 部件中,都要写 CompositionInitializer.SatisfyImports(this);
但这样貌似很麻烦,我希望的是在主程序中只进行一次装配,所有的组件树就都搞定了,但实际测试中发现不行,LZ有什么办法么?

 回复 引用 查看   
#6楼 2012-02-02 14:57 | 袜子不回头      
@FerventDesert
写一个基类 在构造方法写上CompositionInitializer.SatisfyImports(this); 大家都去继承他

 回复 引用 查看   
#7楼 2012-02-02 16:46 | FerventDesert      
@袜子不回头
呵呵,在一个单继承的语言当中使用继承是多么奢侈的事情,有些MEF组件已经从其他类继承了,我还是乖乖的多写那么几行代码吧

 回复 引用 查看   
发表评论

昵称: [登录] [注册]

主页:

邮箱:(仅博主可见)

评论内容:

  登录  注册

[使用Ctrl+Enter键快速提交评论]

0 1789077 w/CYSgNUIK8=
微软最有价值专家(MVP)

微软技术社区精英(CNTAC)

2010年IT博客大赛50强

微软最有影响力开发者(GDI)


Bing Maps开发一群:75662563
微软技术群-重庆站:97035589
RIA技术联盟QQ群:26917590
昵称:Bēniaǒ
园龄:4年6个月
荣誉:推荐博客
粉丝:407
关注:26

随笔分类(285)

文章分类(14)

积分与排名

  • 积分 - 760858
  • 排名 - 60

最新评论