2016-03-27 7 views
8

Ich möchte ein mehrdimensionales Array mit Reflection.Emit erstellen und das Element festlegen. Wie die # -Code folgenden C:So erstellen Sie ein mehrdimensionales Array mit Reflection.Emit

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

Und IL-Code verwandeln sich in:

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) 

Die IL-Code-Array zu erstellen:

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

Und die IL-Code des Arrays Element festlegen :

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

Welche Art von IL Generator.Emit() - Code entspricht diesen beiden IL-Satz?

Antwort

3

können Sie fast übersetzen, dass IL verbal:

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); 

Natürlich müssen Sie Reflektion verwenden tatsächlich die ConstructorInfo und das MethodInfo Objekt erhalten Sie für die Newobj und Call Codes benötigen. Hier

+1

danke, ich schrieb zuerst [,] als [] [] und bekam einen falschen IL-Code, jetzt sehe ich –

+0

'[,]' und '[] []' beziehen sich auf zwei verschiedene Arten von Arrays, siehe [ Diese Frage] (http://stackoverflow.com/questions/597720/what-are-the-differences-between-a-multidimensional-array-and-an-array-of-arrays) für weitere Informationen über die Unterschiede. – poke

3

ist ein Beispiel:

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 

diesem Beispiel wird ein dynamisches Verfahren, die das Array erzeugt, füllt den Wert in [1,1], und dann das Array zurückgibt.

Verwandte Themen