.NET(C#):Emit创建异常处理的方法

时间:2023-11-05 04:03:02 

目录

Emit异常处理流程
显示Exception对象的Message属性
返回目录
Emit异常处理流程
来看这种C#异常处理代码:


        static void doo(Exception e)

        {

            try

            {

                throw e;

            }

            catch (ApplicationException ex)

            {

                Console.WriteLine("捕获ApplicationException");

            }

            catch

            {

                Console.WriteLine("捕获Exception");

            }

            finally

            {

                Console.WriteLine("finally块");

            }

        }


我们将用反射Emit创建一个这样的方法。

其实IL中的异常处理代码还是比较复杂的,你可以在Reflector下看看异常处理的IL代码。不过好在ILGenerator类提供了一些方便的方法来创建异常处理代码。

基本套路就是用如下ILGenerator的方法:

BeginExceptionBlock方法来开始异常处理代码(相当于try)。
之后的代码可以用Opcodes.Throw来抛出异常,或者调用其他可以抛出异常的代码。
接着用BeginCatchBlock方法来开始一个Catch块,该方法可以指定catch需要捕获的异常类型,另外有一点需要注意的是凡是进入该catch方法,逻辑栈上会有相应类型的异常对象。 同时,这里也可以用Opcodes.Rethrow来重新抛出异常。
最后BeginFinallyBlock方法开始一个finally块。 (这里不需要手动加Opcodes.Leave)
当全部异常处理代码写完后,加上EndExceptionBlock方法来结束整块异常处理代码块。
注意方法最后还是必须要加IL的ret指令的(Opcodes.Ret),否则CLR无法运行此方法。

来看代码:


        //+ using System.Reflection;

        //+ using System.Reflection.Emit;

        static void Main(string[] args)
        {
            var dm = GetMethod();

            dm.Invoke(null, new object[] { new ApplicationException() });

            dm.Invoke(null, new object[] { new Exception() });

        }
        static DynamicMethod GetMethod()

        {

            var dm = new DynamicMethod("", null, new Type[] { typeof(Exception) });

            var ilgen = dm.GetILGenerator();

            //try {

            ilgen.BeginExceptionBlock();

            //加载第一个参数,并throw

            ilgen.Emit(OpCodes.Ldarg_0);

            ilgen.Emit(OpCodes.Throw);

            ilgen.BeginCatchBlock(typeof(ApplicationException));

            //清空栈上的异常对象

            ilgen.Emit(OpCodes.Pop);

            ilgen.EmitWriteLine("捕获ApplicationException");

            ilgen.BeginCatchBlock(typeof(Exception));

            //清空栈上的异常对象

            ilgen.Emit(OpCodes.Pop);

            ilgen.EmitWriteLine("捕获Exception");

            ilgen.BeginFinallyBlock();

            ilgen.EmitWriteLine("finally块");

             //结束整个处理块

            ilgen.EndExceptionBlock();

            ilgen.Emit(OpCodes.Ret);

            return dm;

        }

输出:


捕获ApplicationException

finally块

捕获Exception

finally块

返回目录
显示Exception对象的Message属性
上面的代码并没有显示Exception对象的Message属性,上面主要是介绍Emit异常处理的流程,下面来看看怎样显示Message属性,如果是直接输出当然简单了,不过如果用到Console.WriteLine的格式字符串的话,需要在catch代码块中用一个临时变量。

如下代码:


        //+ using System.Reflection;

        //+ using System.Reflection.Emit;
        static void Main(string[] args)
        {
            var dm = GetMethod();

            dm.Invoke(null, new object[] { new Exception("来自Mgen!") });
        }

        static DynamicMethod GetMethod()
        {

            var dm = new DynamicMethod("", null, new Type[] { typeof(Exception) });

            var ilgen = dm.GetILGenerator();

            //try {

            ilgen.BeginExceptionBlock();

            //加载第一个参数,并throw

            ilgen.Emit(OpCodes.Ldarg_0);

            ilgen.Emit(OpCodes.Throw);

            ilgen.BeginCatchBlock(typeof(Exception));

            //临时变量 和 需要的反射信息

            var exp = ilgen.DeclareLocal(typeof(Exception));

            var msg = typeof(Exception).GetProperty("Message").GetGetMethod();

            var output = typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string), typeof(object) });

            //保存异常对象到临时变量exp

            ilgen.Emit(OpCodes.Stloc, exp);

            //格式字符串进栈

            ilgen.Emit(OpCodes.Ldstr, "错误信息: {0}");

            //加载临时变量

            ilgen.Emit(OpCodes.Ldloc, exp);

            //获取Message属性

            ilgen.Emit(OpCodes.Callvirt, msg);

            //调用有格式字符串的Console.WriteLine

            ilgen.Emit(OpCodes.Call, output);

            //结束整个处理块

            ilgen.EndExceptionBlock();

            ilgen.Emit(OpCodes.Ret);

            return dm;

        }

输出:


错误信息: 来自Mgen!

标签:Emit,异常
0
投稿

猜你喜欢

  • Java中Prime算法的原理与实现详解

    2022-06-11 23:16:29
  • Android仿微信实现首字母导航条

    2022-07-16 11:14:29
  • Spring Boot + Mybatis多数据源和动态数据源配置方法

    2023-02-16 17:15:31
  • C# 多线程中经常访问同一资源可能造成哪些问题

    2022-09-08 05:22:47
  • Mybatis打印替换占位符后的完整Sql教程

    2023-11-08 22:56:13
  • gradle使用maven-publish发布jar包上传到私有maven配置

    2022-11-22 07:07:54
  • Spring MVC请求参数接收的全面总结教程

    2023-11-28 19:44:47
  • 基于java实现人机猜拳游戏

    2021-06-09 08:50:32
  • 详解Maven安装教程及是否安装成功

    2021-07-14 00:00:21
  • Java 从互联网上爬邮箱代码示例

    2022-02-27 16:40:57
  • java构造函数示例(构造方法)

    2022-05-08 19:06:03
  • android开发之蜂鸣提示音和震动提示的实现原理与参考代码

    2022-11-22 21:47:38
  • Android列表组件ListView使用详解之动态加载或修改列表数据

    2023-10-01 16:14:38
  • RestTemplate自定义请求失败异常处理示例解析

    2021-12-03 22:13:17
  • Kotlin select使用方法介绍

    2022-05-28 19:34:27
  • C#中面向对象编程机制之继承学习笔记

    2023-12-12 03:46:43
  • Java详解数据类型的定义与使用

    2023-09-29 01:35:20
  • Springboot初始化项目并完成登入注册的全过程

    2023-07-31 15:09:09
  • java日期工具类实例分享

    2023-05-17 17:50:40
  • SpringCloud中的Consul详解

    2023-05-26 04:56:41
  • asp之家 软件编程 m.aspxhome.com