System.Reflection - 如何调用次级类型的getter方法?

时间:2013-07-19 11:58:08

标签: c# .net reflection system.reflection

我有三节课:

public class TestA
{
    public string Str1 { get; set; }
    public string Str2 { get; set; }
    public TestB TestB { get; set; }

    public TestA()
    {
        Str1 = "string1";
        Str2 = "string2";
        TestB = new TestB();
    }
}

public class TestB
{
    public string Str3 { get; set; }
    public string Str4 { get; set; }
    public TestC ObjTestC { get; set; }
    public TestB()
    {
        Str3 = "string3";
        Str4 = "string4";
        ObjTestC = new TestC();
    }
}

public class TestC
{
    public string Str5 { get; set; }
    public TestC()
    {
        Str5 = "string5";
    }
}

现在,我已经获得了所有PropertyInfo并创建了一个新对象:

        //Get all the properties

        var prop = typeof(TestA).GetProperties();

        for (int i = 0; i < prop.Count(); i++)
        {
            var propertyInfo = prop[i];
            if (propertyInfo.PropertyType.Namespace != "System")
            {
                if (propertyInfo.PropertyType.IsGenericType &&
                    propertyInfo.PropertyType.GetGenericTypeDefinition() == typeof(List<>))
                {
                    Type itemType = propertyInfo.PropertyType.GetGenericArguments()[0]; // use this...
                    var listObjectProperties = itemType.GetProperties();
                    prop = prop.Union(listObjectProperties).ToArray();
                }
                else
                {
                    var childProp = propertyInfo.PropertyType.GetProperties();
                    prop = prop.Union(childProp).ToArray();
                }
            }
        }


        //Create Object
        TestA testA = new TestA();

现在,我需要调用每个属性的getter方法。我尝试了以下调用TestA类属性的getter。但是,当尝试在TestB和TestC中调用属性的getter时,它会抛出错误:

        // Loop through all properties
        foreach (PropertyInfo propertyInfo in prop)
        {
            MethodInfo getterMethodInfo = propertyInfo.GetGetMethod();
            var obj=getterMethodInfo.Invoke(testA, null);
        }

请帮忙......

提前致谢

1 个答案:

答案 0 :(得分:7)

检查此代码时,我希望prop数组包含TestATestB上的所有属性,并且根本不会触及TestC

// Gets TestA's properties...
var prop = typeof(TestA).GetProperties();

// Then gets TestB's properties...
var childProp = propertyInfo.PropertyType.GetProperties();
prop = prop.Union(childProp).ToArray();

如果你想从这三种类型中获取属性,你应该递归地进行。每次更改这些对象的结构时,您都不希望必须修改此循环。

但这不是真正的问题。

真正的问题是getterMethodInfo.Invoke(testA, null)是针对来自TestB的属性调用的,但是传递了TestA的实例,因此错误。想想如果不使用反射,你将如何获得这个值。您现有的代码基本上是这样做的:

var value = testA.Str3; // !!! there is no Str3 property on TestA !!!

要正常工作,您需要这样做:

var value = testA.TestB.Str3;

了解我们如何调用一系列属性来获得所需的值?先是TestA.TestB,然后是TestB.Str3。如果你需要使用反射,那么代码需要做同样的事情:

var obj = (object)testA;
obj = testBPropOnTestA.GetValue(obj, null);
var value = str3PropOnTestB.GetValue(obj, null);

因此,您不需要在集合中存储PropertyInfo的简单实例,而是需要存储PropertyInfo s的列表。

所以你的最终“获取值”循环看起来更像是:

foreach (var propertyChain in prop)
{
    var obj = (object)testA;
    foreach (var property in propertyChain)
    {
        obj = property.GetValue(obj, null);
        if (obj == null) {
            break;
        }
    }
    Console.WriteLine("{0} = {1}",
        string.Join(".", propertyChain.Select(x => x.Name)),
        obj);
}

我会留意如何修复第一个“聚集PropertyInfo s”循环给你。