C# - 在复杂的类层次结构中查找具有特定名称属性的所有对象

时间:2016-02-18 15:55:16

标签: c# attributes class-hierarchy

我有一个复杂的类层次结构:类B是类A的属性,列表(类C),类D是类B的属性,等等 - 许多级别的父子关系。 层次结构中的某些类具有字符串属性" foobar"。有些课程没有。 我有一个A类的实例。我需要找到层次结构中具有属性" foobar"的所有对象,并将其值更改为" qwerty"。 在C#中有一种简单的方法吗?

public class ClassD
{
    public string fooBar;
}
public class ClassC 
{ }
public class ClassB
{
    public List<ClassC> classCList;
    public ClassD classDInstance;
    public string fooBar;
}
public class ClassA
{
    public ClassB classBInstance;
}

2 个答案:

答案 0 :(得分:2)

您希望更改具有给定名称的属性的值

这非常难看,但可能会让您知道如何使用reflection完成您想要的任务:

static void Set<T>(dynamic obj, string property, T value) {
    //Iterate through the fields on your object.  Can change this to properties or do both.
    foreach (var field in obj.GetType().GetFields(BindingFlags.Public | BindingFlags.Instance)) {
        var fieldValue = field.GetValue(obj);

        //If we have a collection, then iterate through its elements.
        if (field.FieldType.GetInterface("System.Collections.IEnumerable") != null && !(fieldValue is string) && fieldValue != null)
            foreach (var item in fieldValue) SetField<T>(item, property, value);
        //If field name and type matches, then set.  
        else if (field.Name == property && field.FieldType == typeof(T)) field.SetValue(obj, value);
    }
}

你可以这样称呼它:

Set(obj, "fooBar", "qwerty");

请注意,此目前仅遍历字段,因为这是您的类的设置方式(公共字段,而不是属性)。如果要包含属性,可以将其更改为使用属性,或组合字段和属性并迭代both

同样,我不是说使用这种方法,但可能会给你一些想法。

答案 1 :(得分:2)

你可以使用Reflection,虽然我不是这样的忠实粉丝,但我猜这是完成你需要的最简单方法。

下面的代码不是很漂亮,但希望能给你一个想法。

    private static void replaceFoobar(object obj)
    {
        if (obj == null)
            return;

        const BindingFlags bindingFlags = BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.GetField | BindingFlags.GetProperty | BindingFlags.Instance;
        FieldInfo[] members = obj.GetType().GetFields(bindingFlags);

        if (!members.Any())
            return;

        foreach (var item in members)
        {
            if (item.Name == "fooBar" || item.Name == "<fooBar>k__BackingField")
            {
                item.SetValue(obj, "qwerty");
            }
            else
            {
                object value = item.GetValue(obj);

                if (value != null && value is IEnumerable)
                {
                    foreach (var itemE in ((IEnumerable)value))
                        replaceFoobar(itemE);
                }
                else
                    replaceFoobar(value);
            }
        }
    }