如何在Linq中使用通用接口类型进行过滤?

时间:2014-12-01 16:08:30

标签: c# linq generics

我需要在列表中处理通用接口的所有变体。

在以下示例中,DerivedPerson每个都包含对其PersonTemplate的引用。

internal interface ITemplate
{
    int TemplateID { get; set; }
}

internal class PersonTemplate : ITemplate
{
    public PersonTemplate(int templateId)
    {
        TemplateID = templateId;
    }

    public int TemplateID { get; set; }
}

internal interface ITemplateInstance<TTemplate> where TTemplate : ITemplate
{
    TTemplate TemplateReference { get; set; }
}

internal class DerivedPerson : ITemplateInstance<PersonTemplate>
{
    public DerivedPerson(PersonTemplate template)
    {
        TemplateReference = template;
    }

    public PersonTemplate TemplateReference { get; set; }
}

internal class Program
{
    private static void Main(string[] args)
    {
        IList<object> objects = new List<object>
        {
            new object(),
            new DerivedPerson(new PersonTemplate(1)),
            new DerivedPerson(new PersonTemplate(2))
        };

        //Count = 2
        List<ITemplate> personTemplates = objects
                                            .OfType<ITemplateInstance<PersonTemplate>>()
                                            .Select(pi => pi.TemplateReference as ITemplate)
                                            .ToList();

        //Count = 0 (!)
        List<ITemplate> allTemplates = objects
                                            .OfType<ITemplateInstance<ITemplate>>()
                                            .Select(pi => pi.TemplateReference)
                                            .ToList();
    }
}

Main方法中,我想要处理任何模板的所有实例,而不仅仅是PersonTemplate类型的模板。我正在寻找一种可以处理它的通用方法。

我发现了一个类似且已被接受的问题here。但我的目标不是继续使用已过滤的IList<object>,而是使用IList<ITemplate>

1 个答案:

答案 0 :(得分:4)

你可以这样做:

List<ITemplate> allTemplates
              = objects.Where(o => o.GetType()                                                                              
                                    .GetInterfaces()
                                    .Any(i => i.IsGenericType && i.GetGenericTypeDefinition() == typeof(ITemplateInstance<>)))
                       .Select(o => o.GetType()
                                     .GetProperty("TemplateReference")
                                     .GetValue(o))
                       .OfType<ITemplate>()
                       .ToList();