如何确定MEF插件类的类型?

时间:2012-04-19 09:40:28

标签: c# metadata mef

我使用MEF(C#4.0)作为我的插件系统。有时需要获取所有插件的派生类型列表(主要用于XML反序列化)。这可以在不实例化插件的情况下完成吗?

这有效但需要实例化:

var cat = new DirectoryCatalog(path, "*.dll");
var container = new CompositionContainer(cat);
container.ComposeParts(this);
foreach (var plugin in Plugins)
{
    // Would be better if this could be done via Metadata!
    DoStuff(plugin.Value.GetType());
}
// ...
[ImportMany]
public Lazy<PluginBase, IPluginMetadata>[] Plugins
{
    get;
    private set;
}

问:是否可以通过某些ExportAttribute或其他技术来实现这一目标?

感谢。

2 个答案:

答案 0 :(得分:2)

MEF无法自行提供此信息。要了解原因,请考虑通过属性进行以下导出:

[Export(typeof(PluginBase))]
public PluginBase MyPlugin
{
    get
    {
        if (someCondition)
        {
           return new FooPlugin();
        }
        else
        {
           return new BarPlugin();
        }
    }
}

但是,您仍然可以在导出的元数据中包含该类型(如下所示或通过包含元数据的自定义导出属性):

[Export(typeof(PluginBase))]
[ExportMetadata("Type", typeof(Foo))]
public class Foo : PluginBase
{
}

并添加IPluginMetadata.Type成员。

答案 1 :(得分:1)

你也可以对ComposableParts做一点迭代,假设你有一个包含Catalog的静态MEFHelper类,你可以编写这段代码

public static IEnumerable<Type> GetExportedTypes<T>()
{
    return catalog.Parts
        .Select(part => ComposablePartExportType<T>(part))
        .Where(t => t != null);
}

private static Type ComposablePartExportType<T>(ComposablePartDefinition part)
{

    if (part.ExportDefinitions.Any(
        def => def.Metadata.ContainsKey("ExportTypeIdentity") &&
            def.Metadata["ExportTypeIdentity"].Equals(typeof(T).FullName)))
    {
        return ReflectionModelServices.GetPartType(part).Value;
    }
    return null;
}

我在this blog post解释了这个问题(你可以下载一个工作样本)