Вопрос

Думаю, я спускаю голову. :) Я боролся с этим уже два дня. Код выглядит правильно. Но по какой -то причине, когда я пытаюсь получить доступ к полю [ImportMany], оно является нулевым или, по крайней мере, не возвращать каких -либо значений.

Он получает 3 части в каталоге, но они не применяются к ленивому импорту, который я определяю.

Вот мой код:

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();
        }
    }

}
Это было полезно?

Решение

Ваш атрибут метаданных определяет экспорт как typeof(IUIExtensionDetails) который является вашим контрактом на метаданные, а не ваше фактическое продление. Измените пользовательский конструктор атрибутов на:

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

Другие советы

Ваша ошибка здесь:

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

Вы должны передать тип контракта, а не тип метаданных:

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

(Кроме того, чтобы убедиться, что ваш пользовательский класс экспорта обладает правильными свойствами, как и ожидалось в импорте с метаданными, я бы заставил его реализовать IUIExtensionDetails интерфейс. Но это не обязательно.)

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top