6

我不太了解 MEF,所以希望这是我认为它如何工作的简单修复。

我正在尝试使用 MEF 来获取有关某个类以及应如何使用它的一些信息。我正在使用元数据选项来尝试实现这一目标。我的接口和属性如下所示:

public interface IMyInterface
{
}

public interface IMyInterfaceInfo
{
    Type SomeProperty1 { get; }
    double SomeProperty2 { get; }
    string SomeProperty3 { get; }
}

[MetadataAttribute]
[AttributeUsage(AttributeTargets.Class, AllowMultiple = true)]
public class ExportMyInterfaceAttribute : ExportAttribute, IMyInterfaceInfo
{
    public ExportMyInterfaceAttribute(Type someProperty1, double someProperty2, string someProperty3)
        : base(typeof(IMyInterface))
    {
        SomeProperty1 = someProperty1;
        SomeProperty2 = someProperty2;
        SomeProperty3 = someProperty3;
    }

    public Type SomeProperty1 { get; set; }
    public double SomeProperty2 { get; set; }
    public string SomeProperty3 { get; set; }
}

用该属性修饰的类如下所示:

[ExportMyInterface(typeof(string), 0.1, "whoo data!")]
[ExportMyInterface(typeof(int), 0.4, "asdfasdf!!")]
public class DecoratedClass : IMyInterface
{
}

尝试使用导入的方法如下所示:

private void SomeFunction()
{
    // CompositionContainer is an instance of CompositionContainer
    var myExports = CompositionContainer.GetExports<IMyInterface, IMyInterfaceInfo>();
}

在我的情况下myExports总是空的。在我的 CompositionContainer 中,我的目录中有一个 Part 有两个ExportDefinitions,两者都具有以下内容ContractName:“MyNamespace.IMyInterface”。根据Metadata我的导出,它也正确加载。

如果我删除AllowMultiplesetter 并仅包含一个导出属性,则该myExports变量现在具有单个导出及其加载的元数据。

我究竟做错了什么?

编辑:如果我使用弱类型元数据,我的导出突然得到满足:

var myExports = CompositionContainer.GetExports<IMyInterface, IDictionary<string, object>>();

任何想法为什么?

4

1 回答 1

10

众所周知,MEF 在处理AllowMultiple = true. 对于完整的解释,您可以例如查看here,无论如何,它源于元数据保存在字典中,当 AllowMultiple 为真时,值是数组,并且这样的东西不能映射到您的 IMyInterfaceInfo 上。

这是我使用的解决方法。首先,您的属性应该从 Attribute 派生,而不是从 ExportAttribute 派生:

[MetadataAttribute]
[AttributeUsage(AttributeTargets.Class, AllowMultiple = true)]
public class ExportMyInterfaceAttribute : Attribute, IMyInterfaceInfo
{
    public ExportMyInterfaceAttribute(Type someProperty1, double someProperty2, string someProperty3)

    {
        SomeProperty1 = someProperty1;
        SomeProperty2 = someProperty2;
        SomeProperty3 = someProperty3;
    }

    public Type SomeProperty1 { get; set; }
    public double SomeProperty2 { get; set; }
    public string SomeProperty3 { get; set; }
}

这意味着导出的类应该有 3 个属性,一个标准导出和您的自定义属性:

[Export(typeof(IMyInterface))]
[ExportMyInterface(typeof(string), 0.1, "whoo data!")]
[ExportMyInterface(typeof(int), 0.4, "asdfasdf!!")]
public class DecoratedClass : IMyInterface

然后,您必须为将要导入的元数据定义一个视图。这必须有一个以 IDictionary 作为参数的构造函数。像这样的东西:

public class MyInterfaceInfoView
{
    public IMyInterfaceInfo[] Infos { get; set; }

    public MyInterfaceInfoView(IDictionary<string, object> aDict)
    {
        Type[] p1 = aDict["SomeProperty1"] as Type[];
        double[] p2 = aDict["SomeProperty2"] as double[];
        string[] p3 = aDict["SomeProperty3"] as string[];

        Infos = new ExportMyInterfaceAttribute[p1.Length];
        for (int i = 0; i < Infos.Length; i++)
            Infos[i] = new ExportMyInterfaceAttribute(p1[i], p2[i], p3[i]);
    }
}

现在您应该可以成功调用

var myExports = CompositionContainer.GetExports<IMyInterface, MyInterfaceInfoView>();
于 2012-06-13T07:48:37.927 回答