使用反射更改只读属性

时间:2010-09-14 05:50:52

标签: c# reflection

有可能吗?用反射或任何其他方式?

5 个答案:

答案 0 :(得分:18)

正如其他人所说,如果您需要这样做,那么您将面临一个设计问题。现在,如果你想知道它是否仅仅是为了知道,或者世界上没有别的方法可以做到这一点,那么在一个非常小的helper library和一个扩展方法的帮助下,它确实是可能的。

请考虑以下代码:

class Person {

    int age;
    string name;

    public int Age { get { return age; } }
    public string Name { get { return name; } }
}

// ...

using Mono.Reflection;
using System.Reflection;

// ...

Person person = new Person (27, "jb");
PropertyInfo nameProperty = typeof (Person).GetProperty ("Name");
FieldInfo nameField = nameProperty.GetBackingField ();
nameField.SetValue (person, "jbe");

使用此代码,您可以仅使用属性获取属性的支持字段,并为支持字段分配新值。您可以阅读有关implementation

的更多详细信息

另请注意,它仅适用于简单属性,例如:

public int Age { get { return age; } }

public string Name {
    get { return name; }
    set { name = value; }
}

public double Velocity { get; private set; }

如果您具有自定义代码的复杂属性,则后备字段解析程序将失败。

答案 1 :(得分:16)

作为自动生成的只读属性的非常脏的解决方法:

class MyTestClass
{
    public string MyProperty { get; }

    public MyTestClass( string MyProperty )
    {
        this.MyProperty = MyProperty;
    }
}

您可以通过以下方式修改自动生成的支持字段:

MyTestClass MyClass = new MyTestClass( "Hello" );
FieldInfo MyWriteableField = MyClass.GetType().GetRuntimeFields().Where( a => Regex.IsMatch( a.Name, $"\\A<{nameof( MyClass.MyProperty )}>k__BackingField\\Z" ) ).FirstOrDefault();
MyWriteableField.SetValue( MyClass, "Another new value" );

PS:当您使用.NET版本时&lt; 4.6您可能需要更改一些代码才能工作。 享受!

答案 2 :(得分:12)

Simon Mattes答案的替代方案是

假设你有:

public class MyClass
{
     public int MyNumber {get;}
}

如果用于测试目的,你可以这样做:

var field = typeof(MyClass).GetField("<MyNumber>k__BackingField", BindingFlags.Instance | BindingFlags.NonPublic);
field.SetValue(anIstanceOfMyClass, 3);

答案 3 :(得分:0)

取决于财产。如果它是一个计算属性 - 不,除非你知道它的基础。如果它只是私有字段的访问者,那么您可以尝试修改该字段。

但总的来说,这是一个非常糟糕的主意,因为你可能不知道这会产生什么副作用。

答案 4 :(得分:-1)

PropertyInfo isReadOnly = typeof(System.Collections.Specialized.NameValueCollection).GetProperty("IsReadOnly", BindingFlags.Instance| BindingFlags.NonPublic);
//Remove the readonly property
isReadOnly.SetValue(param, false, null);

//.............put your code here.....

// Set readonly property
isReadOnly.SetValue(param, true, null);