我使用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
或其他技术来实现这一目标?
感谢。
答案 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解释了这个问题(你可以下载一个工作样本)