Можно ли присвоить массив массиву неизвестного типа, если вы знаете имя поля, которое нужно назначить? - PullRequest
2 голосов
/ 02 ноября 2009

Мне нужно присвоить массив полю. Я не знаю тип полей, но у меня есть ссылка на экземпляр и имя поля. Я могу предположить, что массив может быть приведен к типу поля. Можно ли это сделать?

Bas

Edit:

Надеюсь, этот код прояснит, что я пытаюсь сделать, это вызывает исключение в assign:

class MyClass
{
    static void Main()
    {
        MyClass t = new MyClass();
        A a = new A();
        C[] c = new C[] {new B()};
        t.Assign(a, "field", c);
    }

    void Assign(object obj, string field, object[] value)
    {
        // crashes
        obj.GetType().GetField(field).SetValue(obj, value);
    }
}

class A
{
    public B[] field;
}

class B : C { }

class C { }

Ответы [ 4 ]

5 голосов
/ 02 ноября 2009
instance.GetType()
    .GetField("fieldName", BindingFlags.Instance | BindingFlags.NonPublic)
    .SetValue(instance, array);
1 голос
/ 02 ноября 2009

Дополнительные сведения см. На странице отражения в MSDN .

.
0 голосов
/ 03 ноября 2009

Итак, добрая душа показала мне решение, наслаждайтесь:)

using System;
using System.Reflection;

class MyClass
{
    static void Main()
    {
        MyClass t = new MyClass();
        A a = new A();
        C[] c = new C[] {new B()};
        t.Assign(a, "field", c);
    }

    void Assign(object obj, string field, object[] value)
    {
        FieldInfo pinfo = obj.GetType().GetField(field);
        Array array = Array.CreateInstance(pinfo.FieldType.GetElementType(), value.Length);
        value.CopyTo(array, 0);
        pinfo.SetValue(obj, array);
    }
}

class A
{
    public B[] field;
}

class B : C { }

class C { }
0 голосов
/ 03 ноября 2009

Если код, который вызывает это, не является критичным по времени, вы можете просто использовать FieldInfos SetValue ()

obj.GetField(fieldName, BindingFlags.Instance | BindingFlags.NonPublic).SetValue(obj, newFieldValue);

Если код будет вызываться чаще, вы можете динамически скомпилировать делегат-установщик. Это можно сделать, например, с использованием облегченной генерации кода:

Action<TObject, TField> ConstructGetter(string fieldName)
{
    System.Reflection.FieldInfo field = typeof(TObject).GetField(fieldName);
    DynamicMethod method = new DynamicMethod(typeof(TObject).ToString() + ":" + "Set:" + name,
                                             null, new Type[] { typeof(TObject), typeof(TField) }, typeof(TObject));
    ILGenerator generator = method.GetILGenerator();
    generator.Emit(OpCodes.Ldarg_0);
    generator.Emit(OpCodes.Ldarg_1);
    generator.Emit(OpCodes.Stfld, field);
    generator.Emit(OpCodes.Ret);
    return method.CreateDelegate(typeof(Action<TObject, TField>)) as Action<TObject, TField>;
}
...