如何从.NET程序集中读取数组初始化值

Hay*_*ach 6 .net c# reflection il .net-assembly

如何读取值,让我们说:'99'来自包含此代码的程序集

using Sytem; 

public class Class1
{
    public Class1() {
        // array initializer, want to read '99', '100'... from assembly
        var a = new double[,] { { 1, 2, 3 }, { 99, 100, 101 } };
        // ...
    }

}
Run Code Online (Sandbox Code Playgroud)

到目前为止我做了什么

ILDASM中的方法:

.method /*06000001*/ public hidebysig specialname rtspecialname 
        instance void  .ctor() cil managed
// SIG: 20 00 01
{
  // Method begins at RVA 0x2080
  // Code size       29 (0x1d)
  .maxstack  3
  .locals /*11000001*/ init ([0] float64[0...,0...] a)
  .language '{3F5162F8-07C6-11D3-9053-00C04FA302A1}', '{994B45C4-E6E9-11D2-903F-00C04FA302A1}', '{5A869D0B-6611-11D3-BD2A-0000F80849BD}'
// Source File 'c:\Users\heini19\Documents\Visual Studio 2013\Projects\WcfService1\ClassLibrary1\Class1.cs' 
//000005:         public Class1() {
  IL_0000:  /* 02   |                  */ ldarg.0
  IL_0001:  /* 28   | (0A)000011       */ call       instance void [mscorlib/*23000001*/]System.Object/*01000001*/::.ctor() /* 0A000011 */
  IL_0006:  /* 00   |                  */ nop
  IL_0007:  /* 00   |                  */ nop
//000006:             // array initializer, want to read '99', '100' etc.
//000007:             var a = new double[,] { { 1, 2, 3 }, { 99, 100, 101 } };
  IL_0008:  /* 18   |                  */ ldc.i4.2
  IL_0009:  /* 19   |                  */ ldc.i4.3
  IL_000a:  /* 73   | (0A)000012       */ newobj     instance void float64[0...,0...]/*1B000001*/::.ctor(int32,
                                                                                                         int32) /* 0A000012 */
  IL_000f:  /* 25   |                  */ dup
  IL_0010:  /* D0   | (04)000001       */ ldtoken    field valuetype '<PrivateImplementationDetails>{975506E6-7C24-4C2B-8956-C1B9CF8B80C4}'/*02000003*//'__StaticArrayInitTypeSize=48'/*02000004*/ '<PrivateImplementationDetails>{975506E6-7C24-4C2B-8956-C1B9CF8B80C4}'/*02000003*/::'$$method0x6000001-1' /* 04000001 */
  IL_0015:  /* 28   | (0A)000014       */ call       void [mscorlib/*23000001*/]System.Runtime.CompilerServices.RuntimeHelpers/*01000015*/::InitializeArray(class [mscorlib/*23000001*/]System.Array/*01000016*/,
                                                                                                                                                            valuetype [mscorlib/*23000001*/]System.RuntimeFieldHandle/*01000017*/) /* 0A000014 */
  IL_001a:  /* 0A   |                  */ stloc.0
//000008:             // ...
//000009:         }
  IL_001b:  /* 00   |                  */ nop
  IL_001c:  /* 2A   |                  */ ret
} // end of method Class1::.ctor
Run Code Online (Sandbox Code Playgroud)

编译器 使用初始化值<PrivateImplementationDetails>{975506E6-7C24-4C2B-8956-C1B9CF8B80C4}的字段创建结构,$$method0x6000001-1并使用RuntimeHelpers.InitializeArray它来在运行时初始化新数组.C#中定义的原始值似乎存储在字段中并使用字段句柄进行复制?但这些价值观是如何规划的?

必须有一些更好/更简单的方法来从程序集中读取这些常量?

xan*_*tos 7

没有回应

你想要做的就是不可能做到的事情(注意你的例子中的单个案例可能是可行的,但可能无法满足你的全部需求)

<PrivateImplementationDetails>仅在某些情况下由C#编译器创建(似乎超过一定大小的数字数组).对于所有其他情况,完成构造函数中的直接初始化.有关示例,请参阅http://goo.gl/iC6MRv.

请注意,您看到的IL代码并非"写在石头上"......没有规则要求C#编译器如何创建数组初始值设定项.例如Roslyn(我给出的链接是由Roslyn生成的)使用的是PrivateImplementationDetails代替<PrivateImplementationDetails>{975506E6-7C24-4C2B-8956-C1B9CF8B80C4}.谁知道Mono编译器生成了什么代码.

在最一般的情况下,如果不执行构造函数(因此实例化类)并查看获取的对象,则无法知道初始化数组的值.但这显然还有其他问题(类的初始化可能有副作用,或者可能需要参数)