我希望将MEF用于我正在构建的应用程序的插件系统。我希望每个组件都有一个标识符(GUID),我希望能够查找它。但是,在使用导出的部件时,此ID也很有用。
有没有一种方法可以让我的元数据属性包含导出部分的ID以及属性(或方法),而不是让开发人员填写两次或使用反射从属性中找到它?
答案 0 :(得分:7)
它可能是MEF元数据属性和抽象基类的混合体。我将我的插件合约定义为:
public interface IPluginMetadata
{
Guid PluginId { get; }
}
public interface IPlugin : IPluginMetadata
{
void Initialise();
}
我已强制IPlugin
接口也继承了我们的元数据合约IPluginMetadata
。接下来,我们可以创建自定义导出属性:
[AttributeUsage(AttributeTargets.Class, Inherit = true), MetadataAttribute]
public class ExportPluginAttribute : ExportAttribute, IPluginMetadata
{
public ExportPluginAttribute(string pluginId) : base(typeof(IPlugin))
{
if (string.IsNullOrEmpty(pluginId))
throw new ArgumentException("'pluginId' is required.", "pluginId");
PluginId = new Guid(pluginId);
}
public Guid PluginId { get; private set; }
}
您不需要使用元数据合约IPluginMetadata
来装饰导出属性,因为MEF无论如何都会投影属性,但我更喜欢这样做,所以如果我确实对我的元数据合同进行了更改,那么我的导出属性也应该更新。没有伤害,没有犯规。
一旦我们完成了这个,我们就可以定义一个抽象基类来实现我们的插件契约:
public abstract class PluginBase : IPlugin
{
protected PluginBase()
{
var attr = GetType()
.GetCustomAttributes(typeof(ExportPluginAttribute), true)
.Cast<ExportPluginAttribute>()
.SingleOrDefault();
PluginId = (attr == null) ? Guid.Empty : attr.PluginId;
}
public virtual Guid PluginId { get; private set; }
public abstract void Initialise();
}
然后我们可以通过抽象类的构造函数获取自定义属性,并相应地应用该属性。我们可以这样做:
public IPlugin GetPlugin(Guid id)
{
var plugin = container
.GetExports<IPlugin, IPluginMetadata>()
.Where(p => p.Metadata.PluginId == id)
.Select(p => p.Value)
.FirstOrDefault();
return plugin;
}
还有:
[ExportPlugin("BE112EA1-1AA1-4B92-934A-9EA8B90D622C")]
public class MyPlugin : PluginBase
{
public override Initialise()
{
Console.WriteLine(PluginId);
}
}
我们可以看到out PluginId
通过导出的元数据以及插件的属性公开。
该代码全部未经测试,但我希望它能帮助您朝着正确的方向发展。
答案 1 :(得分:0)
将GUID放在常量中,并将其用于属性和元数据:
[Export(typeof(IFoo))]
[ExportMetadata("GUID", _guid)]
public class Foo : IFoo
{
private const string _guid = "abc";
public string Guid { get { return _guid; } }
}
请注意,您无法使用Guid
类型而不是string
,因为const
关键字不允许这样做。