System.Reflection.Emit-If语句生成

本文关键字:语句 Emit-If Reflection System | 更新日期: 2023-09-27 18:28:42

我正在学习如何在C#中使用ILGenerator生成If语句。

这是我的代码:(ilg是ILGenerator)

ilg.Emit(OpCodes.Ldc_I4_1);
Label start = ilg.DefineLabel();
ilg.Emit(OpCodes.Brfalse, start);
ilg.Emit(OpCodes.Ldstr, "Hello");
ilg.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }));
ilg.MarkLabel(start);

一些注意事项:

  • 单独调用Console.WriteLine非常有效
  • 这是我运行生成的exe时遇到的错误:

未处理的异常:System.InvalidProgramException:公共语言运行时检测到无效程序。在Testing.Test.Main(String[])

System.Reflection.Emit-If语句生成

您的代码运行得几乎很好,只是忘记了将OpCodes.Ret放在末尾,请参阅运行的代码:

public static void Main(string[] args)
{
    var dynamicMethod = new DynamicMethod("PrintHello", typeof(void), null);
    var ilGenerator = dynamicMethod.GetILGenerator();
    ilGenerator.Emit(OpCodes.Ldc_I4_1);
    var toEnd = ilGenerator.DefineLabel();
    ilGenerator.Emit(OpCodes.Brfalse, toEnd);
    ilGenerator.Emit(OpCodes.Ldstr, "Hello");
    ilGenerator.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }));
    ilGenerator.MarkLabel(toEnd);
    ilGenerator.Emit(OpCodes.Ret);
    var @delegate = (Action)dynamicMethod.CreateDelegate(typeof(Action));
    @delegate();
}

然后你会得到这个输出

你好

如果你评论这行ilGenerator.Emit(OpCodes.Ret);,你会得到这个异常

System.InvalidProgramException:公共语言运行时检测到一个无效程序。

请记住,C#中的每个方法都必须具有OpCode.Ret

如果你的主方法是空的,比如这个

public static void Main(string[] args)
{
}

生成的IL(在调试模式下,没有优化)将如下

.method public hidebysig static void Main (string[] args) cil managed 
{
    // Method begins at RVA 0x2050
    // Code size 2 (0x2)
    .maxstack 8
    .entrypoint
    IL_0000: nop
    IL_0001: ret
} // end of method Program::Main

生成的IL(在发布模式下,带有优化)将如下

.method public hidebysig static void Main (string[] args) cil managed 
{
    // Method begins at RVA 0x2050
    // Code size 1 (0x1)
    .maxstack 8
    .entrypoint
    IL_0000: ret
} // end of method Program::Main