深度查找类实例中的属性类型

时间:2012-01-16 10:26:41

标签: .net reflection system.reflection

我有一个解析xml的方法,并从该xml创建一个指定类型的对象。 这都是使用泛型完成的,以便为所有类型提供通用方法。

我的问题是我想使用其类型名称(而不是名称)在各种类中搜索属性。 让我们说property属性类型为“type1”,然后在下面声明了一些类定义:

class foo1
{
  type1 prop1{get;set;}
}

class foo2
{
  foo1 prop2{get;set;}
}

class foo3:foo2
{
  type2 prop3{get;set;}
}

对于所有上面声明的类,如果我创建对象,那么我想访问上述类的每个实例的type1类型属性,即我应该能够获得声明为{{1}的属性的值来自type1foo1foo2类的对象。我真的想要一种通用的方法来实现这一点,因为类可能会增加。

1 个答案:

答案 0 :(得分:1)

以下是几乎执行此操作的一种方法。缺少的是使用反射,BindingFlags.FlattenHierarchy不返回父类的私有方法。将这些类型标记为受保护或公开将解决此问题。 (您也可以手动遍历基类以读取私有成员。)

如果要查找声明给定类型属性的程序集中的所有类型,可以编写如下方法:

// using System.Reflection

public IEnumerable<Type> GetTypesWithPropertyOfType(Assembly a, Type t)
{
    BindingFlags propertyBindingFlags = BindingFlags.Public 
                                       | BindingFlags.NonPublic 
                                       | BindingFlags.Instance 
                                       | BindingFlags.FlattenHierarchy;

    // a property is kept if it is assignable from the type
    // parameter passed in            
    MemberFilter mf = (pi, crit)=>
          (pi as PropertyInfo)
          .PropertyType
          .IsAssignableFrom(t);

    // a class is kept if it contains at least one property that
    // passes the property filter.  All public and nonpublic properties of
    // the class, and public and protected properties of the base class,
    // are considered
    Func<Type, bool> ClassFilter = 
        c=>c.FindMembers(MemberTypes.Property, propertyBindingFlags, mf, null)
            .FirstOrDefault() != null;

    // return all classes in the assembly that match ClassFilter
    return
        a.GetTypes()
        .Where(c=>c.IsClass)
        .Where(ClassFilter);
}

要在执行程序集中查找定义或继承类型type1的属性的类,可以调用:

    var v = GetTypesWithPropertyOfType(
           Assembly.GetExecutingAssembly(),
           typeof(type1));

    foreach (var n in v) Console.WriteLine(n.FullName);

打印出foo1。如果定义foo类的代码被修改为(a)使foo1.prop1公开或受保护,(b)使foo2继承自foo1,则上面的代码将打印出来:

foo1
foo2
foo3

正如所料。