2016-03-27 20 views
8

Desidero creare un array multidimensionale utilizzando Reflection.Emit e impostarlo come elemento. Come il seguente codice C#:Come creare un array multidimensionale utilizzando Reflection.Emit

int[,] nums = new int[2, 2]; 
nums[1, 1] = 2; 

e si trasformano in codice IL:

IL_0000: nop 
IL_0001: ldc.i4.2 
IL_0002: ldc.i4.2 
IL_0003: newobj instance void int32[0..., 0...]::.ctor(int32, int32) 
IL_0008: stloc.0 
IL_0009: ldloc.0 
IL_000a: ldc.i4.1 
IL_000b: ldc.i4.1 
IL_000c: ldc.i4.2 
IL_000d: call instance void int32[0..., 0...]::Set(int32, int32, int32) 

Il codice IL per creare array:

newobj instance void int32[0..., 0...]::.ctor(int32, int32) 

E il codice IL per impostare elemento dell'array :

call instance void int32[0..., 0...]::Set(int32, int32, int32) 

Che tipo di codice IL Generator.Emit() corrisponde a queste due frasi IL?

risposta

3

Si può quasi tradurre quella IL verbalmente:

il.Emit(OpCodes.Ldc_I4_2); 
il.Emit(OpCodes.Ldc_I4_2); 

var constructor = typeof(int[,]).GetConstructor(new Type[]{ typeof(int), typeof(int) }); 
il.Emit(OpCodes.Newobj, constructor); 
il.Emit(OpCodes.Stloc_0); 
il.Emit(OpCodes.Ldloc_0); 
il.Emit(OpCodes.Ldc_I4_1); 
il.Emit(OpCodes.Ldc_I4_1); 
il.Emit(OpCodes.Ldc_I4_2); 

var setMethod = typeof(int[,]).GetMethod("Set"); 
il.Emit(OpCodes.Call, setMethod); 

Naturalmente, è necessario utilizzare la reflection per ottenere effettivamente il e l'oggetto ConstructorInfoMethodInfo il necessario per le Newobj e Call codici.

+1

grazie, ho scritto [,] come [] [] all'inizio e ho sbagliato il codice IL, ora vedo –

+0

'[,]' e '[] []' si riferiscono a due diversi tipi di matrici, vedi [ questa domanda] (http://stackoverflow.com/questions/597720/what-are-the-differences-between-a-multidimensional-array-and-an-array-of-arrays) per ulteriori informazioni sulle differenze. – poke

3

Ecco un esempio:

DynamicMethod method = 
    new DynamicMethod("Test" , typeof(int[,]), new Type[]{}); 

var generator = method.GetILGenerator(); 

//get the constructor that takes in 2 integers (the dimensions of the array) 
var constructor = typeof (int[,]) 
    .GetConstructor(new {typeof (int), typeof (int)}); 

//get the Set method that takes in 3 integers; 2 indexes and the value 
var set_method = typeof(int[,]) 
    .GetMethod("Set", new[] { typeof(int), typeof(int), typeof(int) }); 

var local = generator.DeclareLocal(typeof (int[,])); //local variable to reference the array 

generator.Emit(OpCodes.Ldc_I4_2); 
generator.Emit(OpCodes.Ldc_I4_2); 
generator.Emit(OpCodes.Newobj, constructor); //invoke the constructor to create the array 
generator.Emit(OpCodes.Stloc, local); 
generator.Emit(OpCodes.Ldloc, local); 
generator.Emit(OpCodes.Ldc_I4_1); 
generator.Emit(OpCodes.Ldc_I4_1); 
generator.Emit(OpCodes.Ldc_I4_2); 
generator.Emit(OpCodes.Call, set_method); //call the Set method to set the value 
generator.Emit(OpCodes.Ldloc, local); 
generator.Emit(OpCodes.Ret); 

var result_method = (Func<int[,]>)method.CreateDelegate(typeof (Func<int[,]>)); 

var result = result_method(); //returns the array 

Questo esempio crea un metodo dinamico che crea la matrice, riempie il valore in [1,1], e quindi restituisce tale matrice.

Problemi correlati