我有一个解析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}的属性的值来自type1
,foo1
,foo2
类的对象。我真的想要一种通用的方法来实现这一点,因为类可能会增加。
答案 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
正如所料。