MEF自定义属性和懒惰

时间:2011-01-17 09:28:09

标签: mef lazy-evaluation lazy-initialization

我想我会失去理智。 :) 我已经在这两天苦苦挣扎了。代码看起来正确。但由于某些原因,当我尝试访问[ImportMany]字段时,它为null,或者至少不返回任何值。

它获取目录中的3个部分,但它们不适用于我定义的Lazy []导入。

这是我的代码:

using System;
using System.Linq;
using System.ComponentModel.Composition;
using System.ComponentModel.Composition.Hosting;

namespace MefTest
{

// Extension interface and metadata

    public interface IUIExtension
    {
       void DoSomething();
    }

    public interface IUIExtensionDetails
    {
        string Name { get; }
        string Uri { get; }
    }

    [MetadataAttribute]
    [AttributeUsage(AttributeTargets.Class, AllowMultiple=false)]
    public class UIExtensionAttribute : ExportAttribute
    {
        public UIExtensionAttribute() : base(typeof(IUIExtensionDetails)) { }

        public string Name { get; set; }
        public string Uri { get; set; }
    }

        // Extensions

    [UIExtension(Name="Test 01", Uri="http://www.yourmomma.com/")]
    public class Test1Extension : IUIExtension
    {
        public void DoSomething() { }
    }

    [UIExtension(Name = "Test 02", Uri = "http://www.yourdaddy.com/")]
    public class Test2Extension : IUIExtension
    {
        public void DoSomething() { }
    }

    [UIExtension(Name = "Test 03", Uri = "http://www.youruncle.com/")]
    public class Test3Extension : IUIExtension
    {
        public void DoSomething() { }
    }

        // Main program

    public class Program
    {
        static void Main(string[] args)
        {
            Program p = new Program();
            p.Run();
        }

        [ImportMany]
        public Lazy<IUIExtension, IUIExtensionDetails>[] Senders { get; set; }

        public void Run()
        {
            Compose();
        }

        public void Compose()
        {
            var catalog =  new AssemblyCatalog(
                System.Reflection.Assembly.GetExecutingAssembly());
            var container = new CompositionContainer(catalog);

            container.ComposeParts(this);

            // This is always 3
            Console.WriteLine(
                (from g in container.Catalog.Parts select g).Count());
            // This is always 0
            Console.WriteLine(Senders.Length);

            Console.ReadKey();
        }
    }

}

2 个答案:

答案 0 :(得分:3)

您的错误在这里:

public UIExtensionAttribute() : base(typeof(IUIExtensionDetails))

您应该在那里传递合同类型,而不是元数据类型:

public UIExtensionAttribute() : base(typeof(IUIExtension))

(另外,为了确保您的自定义导出类具有导入元数据所需的正确属性,我会使其实现IUIExtensionDetails接口。但这不是强制性的。)

答案 1 :(得分:2)

您的元数据属性将导出定义为typeof(IUIExtensionDetails),这是您的元数据合约,而不是您的实际扩展名。将自定义属性构造函数更改为:

public UIExtensionAttribute() : base(typeof(IUIExtension)) { }