Могу ли я изменить приватное поле, наследуемое только на чтение в С#, используя отражение?

как в java У меня есть:

Class.getSuperClass().getDeclaredFields()

как я могу узнать и установить частное поле из суперкласса?

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

Ответ 1

Да, можно использовать отражение, чтобы установить значение поля readonly после запуска конструктора

var fi = this.GetType()
             .BaseType
             .GetField("_someField", BindingFlags.Instance | BindingFlags.NonPublic);

fi.SetValue(this, 1);

ИЗМЕНИТЬ

Обновлен для просмотра в прямом родительском типе. Это решение, вероятно, будет иметь проблемы, если типы являются общими.

Ответ 2

Этот класс позволит вам сделать это:

http://csharptest.net/browse/src/Library/Reflection/PropertyType.cs

Использование:

new PropertyType(this.GetType(), "_myParentField").SetValue(this, newValue);

BTW, он будет работать в общественных/непубличных полях или свойствах. Для удобства использования вы можете использовать производный класс PropertyValue следующим образом:

new PropertyValue<int>(this,  "_myParentField").Value = newValue;

Ответ 3

Да, вы можете.

Для полей используйте класс FieldInfo. Параметр BindingFlags.NonPublic позволяет вам видеть частные поля.

public class Base
{
    private string _id = "hi";

    public string Id { get { return _id; } }
}

public class Derived : Base
{
    public void changeParentVariable()
    {
        FieldInfo fld = typeof(Base).GetField("_id", BindingFlags.Instance | BindingFlags.NonPublic);
        fld.SetValue(this, "sup");
    }
}

и небольшой тест, чтобы доказать, что он работает:

public static void Run()
{
    var derived = new Derived();
    Console.WriteLine(derived.Id); // prints "hi"
    derived.changeParentVariable();
    Console.WriteLine(derived.Id); // prints "sup"
}

Ответ 4

Как и JaredPar, я сделал следующее:

//to discover the object type
Type groupType = _group.GetType();
//to discover the parent object type
Type bType = groupType.BaseType;
//now I get all field to make sure that I can retrieve the field.
FieldInfo[] idFromBaseType = bType.GetFields(BindingFlags.NonPublic | BindingFlags.Instance);

//And finally I set the values. (for me, the ID is the first element)
idFromBaseType[0].SetValue(_group, 1);

Спасибо всем.