.NET Emit 入门教程:第六部分:IL 指令:2:详解 ILGenerator 辅助方法
前言:
经过前面几大部分的学习,已经掌握了 Emit 的前因后果,今天来详细讲解 Emit 中 IL 的部分内容。
如前文所讲,通过 DynamicMethod(或 MethodBuilder)可获得 ILGenerator 这个用于编写 IL 指令的类,并用它来编写 IL 指令。
本篇主要讲解 ILGenerator 的介绍,以及主要的辅助方法,详细的指令方法,则拆分到下一篇介绍。
下面就开邕它的介绍吧:
1、ILGenerator 介绍
ILGenerator 是.NET 中的一个重要组件,用于动态生成 Intermediate Language(IL)代码。
通过ILGenerator,开发人员可以在运行时创建和修改方法体内的IL指令,实现动态方法的生成和优化。
ILGenerator 提供了一组方法,允许程序员发出各种IL指令,包括加载、存储、运算、流程控制等操作,从而实现对方法体逻辑的灵活控制。
在 .NET 开发中,ILGenerator 通常与 DynamicMethod 类结合使用。
通过 DynamicMethod 创建动态方法对象,然后使用 ILGenerator 在其中生成IL代码。
这种结合使开发人员能够在运行时动态生成高效的代码,应用于一些需要动态生成代码的场景,如动态代理、AOP等。
ILGenerator 的灵活性和强大功能为.NET开发提供了更多可能性和自定义性。
2、ILGenerator 简单示例:
先看一下之前文章提到的代码:
DynamicMethod dynamicMethod = new DynamicMethod("MyMethod", typeof(void), null);
ILGenerator il = dynamicMethod.GetILGenerator();
il.EmitWriteLine("hello world!");
il.Emit(OpCodes.Ret);
从示例代码使用了两类方法:
指令方法:il.Emit(OpCodes.Ret)
辅助方法:il.EmitWriteLine("hello world!")
所有的辅助方法,都是基于指令方法的封装,即用指令也可以实现该方法功能,
但用辅助方法,可以更简单的调用,下面开始介绍辅助方法。
3、ILGenerator 辅助方法:EmitWriteLine
该方法封装好的调用 WriteLine 输出控制台消息,使用它可以简单输出控制台方法,而不用编写 Emit 指令方法。
如果用 Emit 指令,编写是这样的:
var il = methodBuilder.GetILGenerator();
il.Emit(OpCodes.Ldstr, "这是一个示例消息");
il.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }));
il.Emit(OpCodes.Ret);
实现效果对应代码:

3、ILGenerator 辅助方法:异常处理 try catch finally
辅助方法中,提供了关于常用的 try catch finally 的封装方法,可以帮助我们更简单的编写IL方法:
看示例代码:
var il = methodBuilder.GetILGenerator(); il.BeginExceptionBlock();// 开始 try
il.EmitWriteLine("hello world!"); il.BeginCatchBlock(typeof(Exception));// 开始 catch
il.EmitWriteLine("hello world on error!"); il.BeginFinallyBlock();// 开始 finally
il.EmitWriteLine("hello world on finally!"); il.EndExceptionBlock();// 结束 il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Ret);
参照对应生成的代码:

4、ILGenerator 辅助方法:异常抛出
辅助方法中,也提供了一个抛出异常的方法,示例代码:
var il = methodBuilder.GetILGenerator(); il.ThrowException(typeof(Exception)); il.Emit(OpCodes.Ret);
查看对应生成:

但是该辅助方法只能生成抛出异常,没有提供异常的参数。
如果需要更详细的异常抛出,则需要使用指令的方法:
指令方法如:
var il = methodBuilder.GetILGenerator(); il.Emit(OpCodes.Ldstr, "这是一个示例异常消息"); // 创建一个新的 Exception 实例
ConstructorInfo ctor = typeof(Exception).GetConstructor(new Type[] { typeof(string) });
il.Emit(OpCodes.Newobj, ctor); // 使用 ThrowException 方法引发异常
il.Emit(OpCodes.Throw); il.Emit(OpCodes.Ret);
生成的对应代码:

5、ILGenerator 辅助方法:定义变量
辅助方法中,有一个是用来辅助定义变量的。
但是它需要配置 Emit 指令使用,示例代码:
ILGenerator il = methodBuilder.GetILGenerator(); LocalBuilder local = il.DeclareLocal(typeof(string)); // 定义变量 il.Emit(OpCodes.Ldstr,"hello world"); // 加载字符串
il.Emit(OpCodes.Stloc, local); // 将字符串赋值给变量 il.Emit(OpCodes.Ldloc, local); //从变量中 加载值进栈
il.Emit(OpCodes.Ret);//返回(带值)
对应生成的代码:

在这个示例中(为了举例,多了中间的赋值取值的过程):
可以看出,定义的临时变量,都是没有名称的,只有类型。
它可以用来临时存值,需要用到的时候,再将值取出,对应两个 Emit 指令:
赋值:il.Emit(OpCodes.Stloc, local); // 将字符串赋值给变量
取值:il.Emit(OpCodes.Ldloc, local); //从变量中 加载值进栈
5、ILGenerator 辅助方法:定义标签
标签的定义,可以理解为跳转,即现在不常用的 goto 语句所需的的标签。
标签的定义,在 if else 中, switch 中,for 循环中,都会常常使用到标签。
标签的使用分为3步,定义标签、设定标签、跳转到标签。
标签定义:
Label label = il.DefineLabel();
设定标签:
il.MarkLabel(label);
跳转标签:
在 IL(Intermediate Language)中,可以使用以下指令来跳转到标签(Label):
条件跳转指令:
beq:如果两个值相等,则跳转到指定的标签。bge:如果第一个值大于或等于第二个值,则跳转到指定的标签。bgt:如果第一个值大于第二个值,则跳转到指定的标签。ble:如果第一个值小于或等于第二个值,则跳转到指定的标签。blt:如果第一个值小于第二个值,则跳转到指定的标签.bne.un:如果两个无符号整数值不相等,则跳转到指定的标签。brtrue:如果值为 true,则跳转到指定的标签。brfalse:如果值为 false,则跳转到指定的标签。brtrue.s:如果值为 true,则跳转到指定的标签(短格式)。brfalse.s:如果值为 false,则跳转到指定的标签(短格式).
无条件跳转指令:
br:无条件跳转到指定的标签。br.s:短格式的无条件跳转到指定的标签。leave:无条件跳转到 try、filter 或 finally 块的末尾。leave.s:短格式的无条件跳转到 try、filter 或 finally 块的末尾.
比较跳转指令:
bgt.un:如果第一个无符号整数值大于第二个值,则跳转到指定的标签。bge.un:如果第一个无符号整数值大于或等于第二个值,则跳转到指定的标签。blt.un:如果第一个无符号整数值小于第二个值,则跳转到指定的标签。ble.un:如果第一个无符号整数值小于或等于第二个值,则跳转到指定的标签.
其他跳转指令:
switch:根据给定的索引值跳转到不同的标签。brnull:如果值为 null,则跳转到指定的标签。brinst:如果对象是类的实例,则跳转到指定的标签。
这些指令可以帮助控制流程,在特定条件下跳转到指定的标签位置执行相应的代码。
通过合理使用这些跳转指令,可以实现复杂的逻辑控制和条件判断。
总结:
这篇教程总结了.NET Emit 中关于 IL 指令的第六部分,着重介绍了 ILGenerator 辅助方法的详细内容。
ILGenerator 是在动态生成程序集时用来生成 Intermediate Language(IL)指令的一个重要工具。
读者通过本篇文章,可以迅速了解到该教程的主要内容和重点,更好地掌握 ILGenerator 辅助方法的使用及 IL 指令的生成过程。
下一篇,我们将重点讲解 IL 的指令内容。
.NET Emit 入门教程:第六部分:IL 指令:2:详解 ILGenerator 辅助方法的更多相关文章
- Golang入门教程(十三)延迟函数defer详解
前言 大家都知道go语言的defer功能很强大,对于资源管理非常方便,但是如果没用好,也会有陷阱哦.Go 语言中延迟函数 defer 充当着 try...catch 的重任,使用起来也非常简便,然而在 ...
- Docker入门教程(六)另外的15个Docker命令
Docker入门教程(六)另外的15个Docker命令 [编者的话]DockerOne组织翻译了Flux7的Docker入门教程,本文是系列入门教程的第六篇,继续介绍Docker命令.之前的第二篇文章 ...
- 无废话ExtJs 入门教程十六[页面布局:Layout]
无废话ExtJs 入门教程十六[页面布局:Layout] extjs技术交流,欢迎加群(201926085) 首先解释什么是布局: 来自百度词典的官方解释:◎ 布局 bùjú: [distributi ...
- Photoshop入门教程(六):通道
学习心得:当大部分人听到通道.心里可能会有一种很害怕的感觉,因为“通道”并不像“图层”这样易于理解,望而生畏.”通道“的本质其实是存储图片的信息,把一张图片比作一个为网站,那么通道就是网站的后台,存储 ...
- RabbitMQ入门教程(十六):RabbitMQ与Spring集成
原文:RabbitMQ入门教程(十六):RabbitMQ与Spring集成 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https: ...
- ASP.NET MVC 5 学习教程:Edit方法和Edit视图详解
原文 ASP.NET MVC 5 学习教程:Edit方法和Edit视图详解 起飞网 ASP.NET MVC 5 学习教程目录: 添加控制器 添加视图 修改视图和布局页 控制器传递数据给视图 添加模型 ...
- SpringMVC强大的数据绑定(2)——第六章 注解式控制器详解
SpringMVC强大的数据绑定(2)——第六章 注解式控制器详解 博客分类: 跟开涛学SpringMVC 6.6.2.@RequestParam绑定单个请求参数值 @RequestParam用于 ...
- 转:JAVAWEB开发之权限管理(二)——shiro入门详解以及使用方法、shiro认证与shiro授权
原文地址:JAVAWEB开发之权限管理(二)——shiro入门详解以及使用方法.shiro认证与shiro授权 以下是部分内容,具体见原文. shiro介绍 什么是shiro shiro是Apache ...
- SaltStack 入门到精通第二篇:Salt-master配置文件详解
SaltStack 入门到精通第二篇:Salt-master配置文件详解 转自(coocla):http://blog.coocla.org/301.html 原本想要重新翻译salt-mas ...
- 【入门】广电行业DNS、DHCP解决方案详解(三)——DNS部署架构及案
[入门]广电行业DNS.DHCP解决方案详解(三)——DNS部署架构及案 DNS系统部署架构 宽带业务DNS架构 互动业务DNS架构 案例介绍 案例一 案例二 本篇我们将先介绍DNS系统部署架构体系, ...
随机推荐
- letcode-两数相除
题解 设未知数: Br= 125 / 3,拆进行如下拆解: Br = 125 / 3 Br = (29 + 96)/3 Br = 29/3 + (32 * 3) / 3 Br = 29/3 + (2 ...
- C++ 多线程的错误和如何避免(2)
试图 join 一个已经 detach 的线程 如果你已经在某个地方分离了线程,那你不可以在主线程再次 join,这是一个明显的错误 比如: #include <iostream> #in ...
- day01---操作系统安装环境准备
虚拟机安装操作系统步骤 1.新建虚拟主机 2.选择自定义 3.稍后安装操作系统 4.操作系统选择linux 5.选择存放位置 6.cpu和核数选择,默认即可 7.内存分配 8.网络选择 9.控制器类型 ...
- 基于javaweb的服装租赁网站
演示 技术+环境+工具 jdk8+maven.3.2.1+mysql5.7+idea+navicat+spring+springmvc+mybatis+bootstrap+jquery+ajax
- Could not resolve type alias 'com.github.mybatis.helper.page.PageSqlInterceptor'.
报错信息 Could not resolve type alias 'com.github.mybatis.helper.page.PageSqlInterceptor'. Cause: java. ...
- FROM_UNIXTIME函数格式化时间戳日期类型
select FROM_UNIXTIME(bce.daysec_time/1000,'%Y-%m-%d %h:%i:%s') ,bce.* from biz_cvent bce where bce.d ...
- 【Azure App Service for Linux】NodeJS镜像应用启动失败,遇见 RangeError: Incorrect locale information provided
问题描述 在App Service For Linux 中,部署NodeJS应用,应用启动失败. 报错信息为: 2023-08-29T11:21:36.329731566Z RangeError: I ...
- Java //100以内的质数的输出(从2开始,到这个数-1结束为止,都不能被这个数本身整除)+优化
1 //100以内的质数的输出(从2开始,到这个数-1结束为止,都不能被这个数本身整除) 2 boolean isFlag = true; //标识i是否被j除尽,修改其值 3 4 for(int i ...
- electron暴露配置文件(用户可随时修改)
配置文件 一般web前端项目配置文件,写死的放在src/config下,需要打包配置的放在.env文件中.但在electron项目中,如果配置数据更改,需要每次给用户打包升级肯定是行不通的.于是外部配 ...
- 修改校准debain/manjaro的时间时钟
一次重启后发现时间竟然从上午变成了晚上!要问,我是怎么发现在的我就是发现上午的时候我的屏幕夜灯突然出现了 使用命令 sudo rm -f /etc/localtime删除本地时间文件 sudo cp ...