一、演示概述
本示例演示如何使用MEF提供的目录(Catalog)的扩展机制实现可过滤导出部件的自定义目录类。主要是通过继承ComposablePartCatalog基类,并实现接口INotifyComposablePartCatalogChanged来完成的。
相关下载(屏幕录像):http://yunpan.cn/cVkvuUNfuDtTX 访问密码 567d
温馨提示:如果屏幕录像和代码不能正常下载,可站内留言,或发邮件到524130780@QQ.COM
欢迎有兴趣研究.NET相关技术的网友加QQ群:18507443
二、自定义部件目录类Catalog
在MEF中,除了可以使用自身提供的注入AggregateCatalog、AssemblyCatalog、DirectoryCatalog这样的目录类以外,也可以自己定义目录类。
自定义目录类需要继承自ComposablePartCatalog类,并实现接口INotifyComposablePartCatalogChanged即可。如下面所示的代码:
- public class FilteredCatalog : ComposablePartCatalog, INotifyComposablePartCatalogChanged
- {
- #region Private Fields
- private readonly ComposablePartCatalog m_ComposablePartCatalog;
- private readonly INotifyComposablePartCatalogChanged m_NotifyComposablePartCatalogChanged;
- private readonly IQueryable<ComposablePartDefinition> m_Parts;
- #endregion
- #region Constructors
- /// <summary>
- /// 默认构造函数。
- /// </summary>
- /// <param name="composablePartCatalog">包含了所有导出部件的目录Catalog。</param>
- /// <param name="expression">筛选条件表达式。</param>
- public FilteredCatalog(ComposablePartCatalog composablePartCatalog, Expression<Func<ComposablePartDefinition, bool>> expression)
- {
- m_ComposablePartCatalog = composablePartCatalog;
- m_NotifyComposablePartCatalogChanged = composablePartCatalog as INotifyComposablePartCatalogChanged;
- m_Parts = composablePartCatalog.Parts.Where(expression);
- }
- #endregion
- #region INotifyComposablePartCatalogChanged
- /// <summary>
- /// 部件目录Catalog已经改变后触发的事件。
- /// </summary>
- public event EventHandler<ComposablePartCatalogChangeEventArgs> Changed
- {
- add
- {
- if (m_NotifyComposablePartCatalogChanged != null)
- {
- m_NotifyComposablePartCatalogChanged.Changed += value;
- }
- }
- remove
- {
- if (m_NotifyComposablePartCatalogChanged != null)
- {
- m_NotifyComposablePartCatalogChanged.Changed -= value;
- }
- }
- }
- /// <summary>
- /// 部件目录Catalog正在发生改变时触发的事件。
- /// </summary>
- public event EventHandler<ComposablePartCatalogChangeEventArgs> Changing
- {
- add
- {
- if (m_NotifyComposablePartCatalogChanged != null)
- {
- m_NotifyComposablePartCatalogChanged.Changing += value;
- }
- }
- remove
- {
- if (m_NotifyComposablePartCatalogChanged != null)
- {
- m_NotifyComposablePartCatalogChanged.Changing -= value;
- }
- }
- }
- #endregion
- #region ComposablePartCatalog
- /// <summary>
- /// 获取目录中包含的部件定义。经过构造函数中的表达式过滤后,已经是传入目录Catalog对象中的一部分导出部件了。
- /// </summary>
- public override IQueryable<ComposablePartDefinition> Parts
- {
- get { return m_Parts; }
- }
- #endregion
- }
上述代码中概括来说包含如下几点内容:
1、构造函数传递了基础目录Catalog对象,这个目录对象可能包含了很多的导出部件,我们要实现的目录过滤类FilteredCatalog就是基于这个目录进行过滤的,它是个全集,而FilteredCatalog是它的子集。另外一个参数是过滤表达式,过滤条件由调用者来编写,至于内部过滤办法实际还是LINQ提供的Where()方法。
2、对于接口INotifyComposablePartCatalogChanged的实现,实际上是和基础目录Catalog对象的事件关联在一起,即当基础目录对象发生改变时,目录过滤类FilteredCatalog也将会收到相应的通知。
3、重写了基类ComposablePartCatalog的Parts集合属性,该属性返回的就是该目录中包含部件定义,凡是在目录中需要被暴露的部件定义都是通过该集合返回的。因此,上述代码中将过滤后的部件定义通过该属性返回。
定义好了过滤类,接下来就是如何使用它了。
三、使用自定义目录类Catalog
如下代码所示:
- // 获取所需的部件。
- DirectoryCatalog catalog = new DirectoryCatalog("controls");
- CompositionContainer container = new CompositionContainer(catalog);
- // 过滤Catalog,生成子组合容器。
- FilteredCatalog filteredCatalog = new FilteredCatalog(catalog,
- o=>o.Metadata.ContainsKey("UC") && o.Metadata["UC"].ToString() == "BB");
- CompositionContainer filteredContainer = new CompositionContainer(filteredCatalog, container);
- UserControl userControl = filteredContainer.GetExportedValue<UserControl>();
- this.MainContentControl.Content = userControl;
首先通过DirectoryCatalog类获取到应用程序根目录下controls子文件夹中的所有部件定义,并以此生成顶级组合容器container(类型为CompositionContainer)。
然后使用自定义目录过滤类FilteredCatalog对DirectoryCatalog目录中的部件定义进行过滤,并生成子组合容器filteredContainer(类型为CompositionContainer)。
最后通过组合容器的GetExportedValue<T>()方法获取指定协议类型的导出部件。需要说明的是,如果组合容器中没有对应协议类型的导出部件则会引发异常。
可通过如下地址获取完整的示例代码和屏幕录像文件。
四、相关资源
1、MSDN官方资料:http://msdn.microsoft.com/zh-cn/library/dd460648(v=vs.110).aspx
2、参考了微软MVP Bēniaǒ的文章《MEF程序设计指南七:使用目录(Catalog)动态装载xap与目录筛选(Filtered Catalog)》,访问地址:http://www.cnblogs.com/beniao/archive/2010/07/26/1782622.html