MSIL实用指南-生成if...else...语句
if...else...语句是非常重要的选择语句,它的生成一般需要ILGenerator的DefineLabel方法和
MarkLabel方法,以及Brtrue_S和Br_S指令。
一、DefineLabel方法
简单调用ILGenerator的DefineLabel方法会得到一个Label结构体。Label结构体没有复杂的功能,
只是简单做一个标签。
实例代码:
Label label1 = ilGenerator.DefineLabel();
二、MarkLabel方法
MarkLabel方法的参数只有一个,就是Label结构体。
用DefineLabel方法得到的Label结构体并没有指定它的位置,为了指定这个标签的位置,必须要调用、
LGenerator的MarkLabel方法。
实例代码:
ilGenerator.MarkLabel(label1);
三、Brtrue_S和Brtrue指令
Brtrue指令的意思是,如果栈上的值是 true 、不为null、或非零值,当前执行顺序则会跳转到指定的标签
上。
Brtrue_S的意思和Brtrue一样,只是它跳转的标签是短格式。
一般一段程序内的标签都没有超过255个,所以都用Brtrue_S指令。
实例代码:
ilGenerator.Emit(OpCodes.Br_S, labelEnd);
四、Br和Br_S指令
Br指令是无条件跳转到指定标签,Br_S指令和它一样,只是跳转标签是短格式的。
和上面一样,一般用Br_S指令
五、生成步骤
public static void Test(bool flag, bool flag2)
{
if (flag)
{
Console.WriteLine("a");
}
else if (flag2)
{
Console.WriteLine("b");
}
else
{
Console.WriteLine("else");
}
}
我们要实现生成上面的代码,需要进行如下步骤。
1.确定label的数量
if...else语句块总会有结束的地方,所以首先要有一个结束标签。
语句块有多少个判断条件,就需要多少个标签;上面有两个逻辑判断条件,所以还要加两个标签。
else语句部分不需要标签。
上面的语句需要三个标签,要先写下声明标签的程序
Label label1 = ilGenerator.DefineLabel();
Label label2 = ilGenerator.DefineLabel();
Label labelEnd = ilGenerator.DefineLabel();
2.生成判断条件
"if"和"else"buf都有判断条件,首先生成它们的判断条件,再生成Brtrue_S指令。
注意它们的判断结果是false才跳转,如果为true不会跳转。if表达式的结果是和
false比较的。
比如第一个if生成是
ilGenerator.Emit(OpCodes.Ldarg_0);
ilGenerator.Emit(OpCodes.Ldc_I4_0);
ilGenerator.Emit(OpCodes.Ceq);
ilGenerator.Emit(OpCodes.Stloc_0);
ilGenerator.Emit(OpCodes.Ldloc_0); ilGenerator.Emit(OpCodes.Brtrue_S, label1);
3.生成条件为true时的语句块
在Brtrue_S时指令后面是条件为true的语句,生成它之后要跳到语句块结束标签,
之后依次给非结束标签指定位置。
例如
ilGenerator.Emit(OpCodes.Ldstr,"a");
ilGenerator.Emit(OpCodes.Call, writeStringLineMethod); ilGenerator.Emit(OpCodes.Br_S, labelEnd);
ilGenerator.MarkLabel(label1);
4.else部分
else是语句块的最后部分,是上面条件都为false才执行的,之后语句块就结束了,所以不需要跳转。
完整的程序如下:
using System;
using System.Reflection;
using System.Reflection.Emit; namespace LX1_ILDemo
{
class Demo23_IfElse
{
static string binaryName = "Demo23_IfElse.exe";
static string namespaceName = "LX1_ILDemo";
static string typeName = "DemoIfElse"; static AssemblyBuilder assemblyBuilder;
static ModuleBuilder moduleBuilder;
static TypeBuilder typeBuilder;
static MethodBuilder mainMethod;
static MethodBuilder testMethod; static void Emit_TestMethod()
{
testMethod = typeBuilder.DefineMethod("Test", MethodAttributes.Public
| MethodAttributes.Static, typeof(void), new Type[] { typeof(bool), typeof(bool) });
ILGenerator ilGenerator = testMethod.GetILGenerator(); Label label1 = ilGenerator.DefineLabel();
Label label2 = ilGenerator.DefineLabel();
Label labelEnd = ilGenerator.DefineLabel();
MethodInfo writeStringLineMethod = typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }); LocalBuilder local1 = ilGenerator.DeclareLocal(typeof(bool));
//if(a)
ilGenerator.Emit(OpCodes.Ldarg_0);
ilGenerator.Emit(OpCodes.Ldc_I4_0);
ilGenerator.Emit(OpCodes.Ceq);
ilGenerator.Emit(OpCodes.Stloc_0);
ilGenerator.Emit(OpCodes.Ldloc_0);
ilGenerator.Emit(OpCodes.Brtrue_S, label1);
ilGenerator.Emit(OpCodes.Ldstr,"a");
ilGenerator.Emit(OpCodes.Call, writeStringLineMethod);
ilGenerator.Emit(OpCodes.Br_S, labelEnd);
ilGenerator.MarkLabel(label1);
//else if(b)
ilGenerator.Emit(OpCodes.Ldarg_1);
ilGenerator.Emit(OpCodes.Ldc_I4_0);
ilGenerator.Emit(OpCodes.Ceq);
ilGenerator.Emit(OpCodes.Stloc_0);
ilGenerator.Emit(OpCodes.Ldloc_0);
ilGenerator.Emit(OpCodes.Brtrue_S, label2);
ilGenerator.Emit(OpCodes.Ldstr, "b");
ilGenerator.Emit(OpCodes.Call, writeStringLineMethod);
ilGenerator.Emit(OpCodes.Br_S, labelEnd);
ilGenerator.MarkLabel(label2);
//else
ilGenerator.Emit(OpCodes.Ldstr, "else");
ilGenerator.Emit(OpCodes.Call, writeStringLineMethod);
ilGenerator.MarkLabel(labelEnd);
ilGenerator.Emit(OpCodes.Ret);
} public static void Generate()
{
InitAssembly();
typeBuilder = moduleBuilder.DefineType(namespaceName + "." + typeName, TypeAttributes.Public); Emit_TestMethod();
GenerateMain(); assemblyBuilder.SetEntryPoint(mainMethod, PEFileKinds.ConsoleApplication);
SaveAssembly();
Console.WriteLine("生成成功");
} static void GenerateMain()
{
mainMethod = typeBuilder.DefineMethod("Main", MethodAttributes.Public
| MethodAttributes.Static, typeof(void), new Type[] { });
var ilGenerator = mainMethod.GetILGenerator();
ilGenerator.Emit(OpCodes.Ldc_I4_0);
ilGenerator.Emit(OpCodes.Ldc_I4_0);
ilGenerator.Emit(OpCodes.Call, testMethod); MethodInfo readKeyMethod = typeof(Console).GetMethod("ReadKey", new Type[] { });
ilGenerator.Emit(OpCodes.Call, readKeyMethod);
ilGenerator.Emit(OpCodes.Pop); ilGenerator.Emit(OpCodes.Ret);
} static void InitAssembly()
{
AssemblyName assemblyName = new AssemblyName(namespaceName);
assemblyBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.RunAndSave);
moduleBuilder = assemblyBuilder.DefineDynamicModule(assemblyName.Name, binaryName);
} static void SaveAssembly()
{
Type t = typeBuilder.CreateType(); //完成Type,这是必须的
assemblyBuilder.Save(binaryName);
}
}
}
MSIL实用指南-生成if...else...语句的更多相关文章
- MSIL实用指南-生成索引器
MSIL实用指南-生成索引器 索引器是一种特殊的属性,它有参数的,也有get和set方法,属性名称一般是"Item",并且方法名称一般名称是"get_Item" ...
- MSIL实用指南-生成foreach语句
foreach可以迭代数组或者一个集合对象.foreach语句格式是它的生成步骤是foreach (<成员> in <集合>) <循环体> 一.声明三个变量,loc ...
- MSIL实用指南-生成for语句
for语句格式是这样的for(<初始化语句>;<条件语句>;<自增减语句>) <循环体> 它可以转换为while语句 if(<条件语句>){ ...
- MSIL实用指南-生成while语句
本篇讲解怎样生成while语句.while语句是编程语言中很重要的循环语句,它的结构是while(<表达式>) <语句或语句块> 当表达式的结果为true时就一直执行语句或语句 ...
- MSIL实用指南-生成异常处理
本篇讲解怎么生成异常.C# 异常处理时建立在四个关键词之上的:try.catch.finally 和 throw. 一.异常的抛出抛出异常在C#语言中要使用throw关键字,使用方法是throw &l ...
- MSIL实用指南-生成属性
本篇讲解怎么生成属性,包括get和set方法. 第一步,生成一个字段生成字段用TypeBuilder.DefineField方法.实例代码: FieldBuilder customerNameBldr ...
- MSIL实用指南-生成构造函数
本篇讲解生成构造函数的一些知识,包括创建实例构造函数.静态构造函数.调用父类构造函数. 生成构造函数的方法生成构造函数的方法是TypeBuilder.DefineConstructor(MethodA ...
- MSIL实用指南-生成接口
本篇讲解怎么样生成接口,即interface. 一.创建类型创建一个接口类型依旧用ModuleBuilder的DefineType方法,但是它的第二个参数必须要有TypeAttributes.Inte ...
- MSIL实用指南-生成内部类
生成内部类用TypeBuilder的DefineNestedType方法,得到另一个TypeBuilder.内部类的可访问性都是TypeAttributes的“Nested”开头一些成员.实例代码:y ...
随机推荐
- SQL Server将自己的查询结果作为待查询数据子列之一
嵌套子查询是SQL语句中比较常用的一种查询方法,开发过程中遇到查询需要将自己的某列作为待查询的数据,在参考别人的SQL语句的写法终于实现了自己需要的功能. 查询语句如下: SELECT DISTINC ...
- 平衡树(Splay)模板
支持区间操作. 单点操作和区间操作分开使用,需要一起使用需要部分修改. 对应题目FJUTOJ2490 #include<cstdio> #include<cstring> #i ...
- python自动拉取备份压缩包并删除3天前的旧备份
业务场景,异地机房自动拉取已备份好的tar.gz数据库压缩包,并且只保留3天内的压缩包文件,用python实现 #!/usr/bin/env python import requests,time,o ...
- windows xp/7/8/8.1/10安全模式详解和系统修复讲解
如果你的电脑因为各种原因(强关电脑等原因导致的电脑蓝屏,或者引导文件受损等)开不了机,进不了电脑桌面,那么就可以看看我这篇文章了. 先贴上百度百科,说得挺好的.最下面有我贴的图片操作过程详解. 百度百 ...
- 笔记:Hibernate 拦截器和事件
Hibernate 在执行持久化的过程中,应用程序通常无法参与其中,通过事件框架,Hibernate 允许应用程序能响应特定的内部事件,从而允许实现某些通用的功能,或者对 Hibernate 进行扩展 ...
- 前端的UI设计与交互之导航篇
在广义上,任何告知用户他在哪里,他能去什么地方以及如何到达那里的方式,都可以称之为导航.当设计者使用导航或者自定义一些导航结构时,请注意:尽可能提供标识.上下文线索,避免用户迷路:保持导航样式和行为一 ...
- 使用jQuery动态克隆表格,并且添加至div中(使用前需要引入jQuery)
<!DOCTYPE html> <html> <head> <title></title> <meta charset="u ...
- PO BO VO DTO POJO DAO DO
PO BO DTO VO 归在一起叫是POJO,简单java对象:DAO 是进行数据库增删改查的类,DO不确定有没有. 重点说下POJO PO 持久对象,数据: BO 业务对象,封装对象.复杂对象 , ...
- Aizu - 0531 Paint Color
白书例题,直接用书上的暴力压缩坐标是可以的,但是看了别人的博客的写法,大概是理解了思想但是看不懂为什么那么压缩,先放这,等明白了补上 #define debug #include<stdio.h ...
- Java基础学习笔记七 Java基础语法之继承和抽象类
继承 继承的概念 在现实生活中,继承一般指的是子女继承父辈的财产.在程序中,继承描述的是事物之间的所属关系,通过继承可以使多种事物之间形成一种关系体系. 例如公司中的研发部员工和维护部员工都属于员工, ...