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

 更新时间:2013年04月24日 14:34:21   作者:  
.NET(C#):Emit创建异常处理的方法,需要的朋友可以参考一下

目录

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!

相关文章

  • Unity向量按照某一点进行旋转

    Unity向量按照某一点进行旋转

    这篇文章主要为大家详细介绍了Unity向量按照某一点进行旋转,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-01-01
  • C# SendMail发送邮件功能实现

    C# SendMail发送邮件功能实现

    这篇文章主要为大家详细介绍了C# SendMail发送邮件功能实现,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-06-06
  • C#基础知识之base关键字介绍

    C#基础知识之base关键字介绍

    本文主要介绍base关键字的使用方法,base关键字可以调用基类重写的方法,可以调用基类的构造方法,还可以在EntityFramework中使用,下面一一介绍。
    2016-04-04
  • C#中WebBrowser.DocumentCompleted事件多次调用问题解决方法

    C#中WebBrowser.DocumentCompleted事件多次调用问题解决方法

    这篇文章主要介绍了C#中WebBrowser.DocumentCompleted事件多次调用问题解决方法,本文讲解了3种情况和各自情况的解决方法,需要的朋友可以参考下
    2015-01-01
  • C# BackgroundWorker组件学习入门介绍

    C# BackgroundWorker组件学习入门介绍

    一个程序中需要进行大量的运算,并且需要在运算过程中支持用户一定的交互,为了获得更好的用户体验,使用BackgroundWorker来完成这一功能
    2013-10-10
  • C# 注册表 操作实现代码

    C# 注册表 操作实现代码

    Windows 操作系统的注册表包含了很多有关计算机运行的配置方式,打开注册表我们可以看到注册表是按类似于目录的树结构组织的
    2009-07-07
  • Task提高异步执行效率技巧

    Task提高异步执行效率技巧

    这篇文章介绍了Task提高异步执行效率的技巧,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-04-04
  • 详解C#的设计模式编程之抽象工厂模式的应用

    详解C#的设计模式编程之抽象工厂模式的应用

    这篇文章主要介绍了C#的设计模式编程之抽象工厂模式的应用,注意区分一下简单工厂模式、工厂方法模式和抽象工厂模式概念之间的区别,需要的朋友可以参考下
    2016-02-02
  • 深入理解C#指针之美

    深入理解C#指针之美

    在C#中,有时候希望通过指针来操作内存,这样可以提高效率。我们可以用unsafe关键字修饰含有指针操作的程序段,感兴趣的小伙伴可以参考一下,希望可以帮到你
    2021-07-07
  • C#循环与循环控制的表达式树实现

    C#循环与循环控制的表达式树实现

    这篇文章介绍了C#循环与循环控制的表达式树实现,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-01-01

最新评论