如何将IL翻译成Emit语句?

时间:2015-04-09 03:33:29

标签: c# reflection.emit

这是C#代码:

public class Calc1 : ICalculator
{
    public int Calculate(int x, int y)
    {
        return x + y;
    }
}

这是IL:

.method public hidebysig newslot virtual final 
    instance int32  Calculate(int32 x,
                              int32 y) cil managed
{
  // Code size       9 (0x9)
  .maxstack  2
  .locals init ([0] int32 CS$1$0000)
  IL_0000:  nop
  IL_0001:  ldarg.1
  IL_0002:  ldarg.2
  IL_0003:  add
  IL_0004:  stloc.0
  IL_0005:  br.s       IL_0007
  IL_0007:  ldloc.0
  IL_0008:  ret
} // end of method Calc1::Calculate

如何将上述IL翻译为Emit语句? 这是我尝试逐个翻译行,但得到了'System.InvalidProgramException'。

        string methodName = "Calculate";

        MethodBuilder getFieldMethod = typeBuilder.DefineMethod(methodName, MethodAttributes.Public | MethodAttributes.Virtual, typeof(int), new Type[] { typeof(int), typeof(int) });
        ILGenerator methodIL = getFieldMethod.GetILGenerator();

        Label iL0007Label = methodIL.DefineLabel();

        methodIL.Emit(OpCodes.Nop);
        methodIL.Emit(OpCodes.Ldarg_1);
        methodIL.Emit(OpCodes.Ldarg_2);
        methodIL.Emit(OpCodes.Add);
        methodIL.Emit(OpCodes.Stloc_0);
        methodIL.Emit(OpCodes.Br_S, iL0007Label);

        methodIL.MarkLabel(iL0007Label);
        methodIL.Emit(OpCodes.Ldloc_0);
        methodIL.Emit(OpCodes.Ret);

我的Emit声明有什么问题?

2 个答案:

答案 0 :(得分:2)

您忘记声明局部变量:

ILGenerator methodIL = getFieldMethod.GetILGenerator();

methodIL.DeclareLocal(typeof(int)); // THIS ONE!

Label iL0007Label = methodIL.DefineLabel();

从技术上讲,您忘记了MethodAttributes.Final,但我不知道您是否要添加它。

请注意,您获得的操作码可能用于调试版本。如果你想查看操作码,我建议使用使用较少操作码的发布版本。

ILGenerator methodIL = getFieldMethod.GetILGenerator();
methodIL.Emit(OpCodes.Ldarg_1);
methodIL.Emit(OpCodes.Ldarg_2);
methodIL.Emit(OpCodes.Add);
methodIL.Emit(OpCodes.Ret);

足以满足您的需求,甚至不使用局部变量。

答案 1 :(得分:0)

我没有太多关于Reflection.Emit的工作,但是当我经常这样做时,我曾经在VS命令提示符中使用PEVerify命令。它告诉我们生成的程序集有什么问题。希望它有所帮助。