如何使用反射在C#中获取auto属性的默认值?

Bil*_*ani 1 .net c# reflection

所以我有这门课

public class MyClass
{
    public int Id {get; set;} = 5
}
Run Code Online (Sandbox Code Playgroud)

使用反射如何获得5

我试图避免创建MyClass的实例.这似乎没有必要.

编辑:它被称为"自动属性初始化程序"值

来自:http://geekswithblogs.net/WinAZ/archive/2015/06/30/whatrsquos-new-in-c-6.0-auto-property-initializers.aspx

ang*_*son 8

该类实际编译为:

public class MyClass
{
    private int MyClass___1<Id> = 5;
    public int Id
    {
        get { return MyClass___1<Id>; }
        set { MyClass___1<Id> = value; }
    }
}
Run Code Online (Sandbox Code Playgroud)

反过来又相当于:

public class MyClass
{
    private int MyClass___1<Id>;

    public MyClass()
    {
        MyClass___1<Id> = 5;
    }

    public int Id
    {
        get { return MyClass___1<Id>; }
        set { MyClass___1<Id> = value; }
    }
}
Run Code Online (Sandbox Code Playgroud)

因此,除了反编译构造函数或构造类型的实例之外,您无法获得该值.

如果您是该类型的作者,最好的方法是将[DefaultValue(...)]属性应用于属性,可以通过反射读取.许多序列化库也将使用此值,以避免序列化具有默认值的属性.

你的课程看起来像这样:

public class MyClass
{
    [DefaultValue(5)]
    public int Id { get; set; } = 5;
}
Run Code Online (Sandbox Code Playgroud)

话虽如此,这是一个演示Mono.Cecil的小型LINQPad程序:

void Main()
{
    var assembly = AssemblyDefinition.ReadAssembly(GetType().Assembly.Location);
    var myClassType =
        (from module in assembly.Modules
         from type in module.Types
         where type.Name == "UserQuery"
         from nestedType in type.NestedTypes
         where nestedType.Name == "MyClass"
         select nestedType).FirstOrDefault();

    var ctor =
        (from method in myClassType.Methods
         where method.IsConstructor
         select method).FirstOrDefault();

    foreach (var instruction in ctor.Body.Instructions)
        Console.WriteLine(instruction.ToString());
}

public class MyClass
{
    public int Id { get; set; } = 5;
}
Run Code Online (Sandbox Code Playgroud)

输出:

IL_0000: ldarg.0
IL_0001: ldc.i4.5
IL_0002: stfld System.Int32 UserQuery/MyClass::<Id>k__BackingField
IL_0007: ldarg.0
IL_0008: call System.Void System.Object::.ctor()
IL_000d: ret
Run Code Online (Sandbox Code Playgroud)

因此在这种情况下可能是可行的 ,但是其他类型的常量等等,就像调用静态方法一样,并不容易处理.

简而言之,创建实例!


Mon*_*Zhu 5

我试图避免创建 MyClass 的实例。好像没必要。

其实不然。如果不调用构造函数,就不会初始化变量Id。但更重要的是,没有实例就没有内存分配,因此您根本无法获得任何价值。

使用反射如何获得 5 ?

但是,如果您确实创建了一个实例,则可以通过以下方式获得它:

Console.WriteLine(typeof(MyClass).GetProperty("Id").GetValue(new MyClass()));
Run Code Online (Sandbox Code Playgroud)