2016-03-27 2 views
8

Я хочу создать многомерный массив, используя Reflection.Emit и установить его. Как следующий C# код:Как создать многомерный массив с помощью Reflection.Emit

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

И превращаются в код 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) 

Ил-код для создания массива:

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

И код IL установить элемент массива за :

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

Какой код IL Generator.Emit() соответствует этим двум предложениям IL?

ответ

3

Вы можете почти перевести, что IL словесно:

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

Конечно, вам нужно использовать отражение на самом деле получить ConstructorInfo и MethodInfo вам объект для Newobj и Call кодов.

+1

Благодарю вас, я написал [,] как [] [] поначалу и получил неправильный код IL, теперь я вижу –

+0

'[,]' и '[] []' ссылаются на два разных типа массивов, см. [ этот вопрос] (http://stackoverflow.com/questions/597720/what-are-the-differences-between-a-multidimensional-array-and-an-array-of-arrays) для получения дополнительной информации о различиях. – poke

3

Вот пример:

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 

В этом примере создается динамический метод, который создает массив, заполняет значение в [1,1], а затем возвращает этот массив.

Смежные вопросы