.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!

(0)

相关推荐

  • Asp.net Mvc 身份验证、异常处理、权限验证(拦截器)实现代码

    1.用户登录 验证用户是否登录成功步骤直接忽略,用户登录成功后怎么保存当前用户登录信息(session,cookie),本文介绍的是身份验证(其实就是基于cookie)的,下面看看代码. 引入命名空间 using System.Web.Security; 复制代码 代码如下: Users ModelUser = new Users() { ID = 10000, Name = UserName, UserName = UserName, PassWord = PassWord, Roles =

  • 关于.NET异常处理的思考总结

    年关将至,对于大部分程序员来说,马上就可以闲下来一段时间了,然而在这个闲暇的时间里,唯有争论哪门语言更好可以消磨时光,估计最近会有很多关于java与.net的博文出现,我表示要作为一个吃瓜群众,静静的看着大佬们发表心情. 以上的废话说的够多了,这里就不再废话了,还是切入正题吧. 在项目开发中,对于系统和代码的稳定性和容错性都是有对应的要求.实际开发项目中的代码与样例代码的区别,更多的是在代码的运行的稳定性.容错性.扩展性的比较.因为对于实现一个功能来说,实现功能的核心代码是一样的,可能只是在写法

  • ASP.NET MVC下基于异常处理的完整解决方案总结

    EntLib的异常处理应用块(Exception Handling Application Block)是一个不错的异常处理框架,它使我们可以采用配置的方式来定义异常处理策略.而ASP.NET MVC是一个极具可扩展开发框架,在这篇文章中我将通过它的扩展实现与EntLib的集成,并提供一个完整的解决异常处理解决方案. 一.基本异常处理策略 我们首先来讨论我们的解决方案具体采用的异常处理策略: 对于执行Controller的某个Action方法抛出的异常,我们会按照指定配置策略进行处理.我们可以采

  • 总结十条.NET异常处理建议

    .NET中从始至终要紧记异常处理的策略:抛出具体的一个异常,而不是只抛出Exception类型的异常,这样能方便我们捕获对应类型的异常.我们在编写代码时要注意考虑到应用程序最差的情况:显示有好的信息,并提供适当的管理员联系信息. 1.不要抛出"new Exception()" 请别这样做.Exception是一个非常抽象的异常类,捕获这类异常通常会产生很多负面影响.通常情况下应该定义我们自己的异常类,并且需要区分系统抛出的异常和我们自己抛出的异常. 2.不要将重要的异常信息存储在Mes

  • .NET中的异常和异常处理用法分析

    本文较为详细的分析了.NET中的异常和异常处理用法.分享给大家供大家参考.具体分析如下: .NET中的异常(Exception) .net中的中异常的父类是Exception,大多数异常一般继承自Exception. 可以通过编写一个继承自Exception的类的方式,自定义异常类! 异常处理机制: 复制代码 代码如下: Try {     //可能发生异常的代码     //后续代码     } //Try以外的代码 catch(Exception e) { } finally { } 上述代

  • ASP.NET mvc异常处理的方法示例介绍

    1.首先常见保存异常的类(就是将异常信息写入到文件中去) 复制代码 代码如下: public class LogManager { private string logFilePath = string.Empty; public LogManager(string logFilePath) { this.logFilePath = logFilePath; FileInfo file = new FileInfo(logFilePath); if (!file.Exists) { file.C

  • ASP.NET MVC异常处理模块详解

    一.前言 异常处理是每个系统必不可少的一个重要部分,它可以让我们的程序在发生错误时友好地提示.记录错误信息,更重要的是不破坏正常的数据和影响系统运行.异常处理应该是一个横切点,所谓横切点就是各个部分都会使用到它,无论是分层中的哪一个层,还是具体的哪个业务逻辑模块,所关注的都是一样的.所以,横切关注点我们会统一在一个地方进行处理.无论是MVC还是WebForm都提供了这样实现,让我们可以集中处理异常. 在MVC中,在FilterConfig中,已经默认帮我们注册了一个HandleErrorAttr

  • 详谈.NET的异常处理

    年关将至,对于大部分程序员来说,马上就可以闲下来一段时间了,然而在这个闲暇的时间里,唯有争论哪门语言更好可以消磨时光,估计最近会有很多关于java与.net的博文出现,我表示要作为一个吃瓜群众,静静的看着大佬们发表心情. 以上的废话说的够多了,这里就不再废话了,还是切入正题吧. 在项目开发中,对于系统和代码的稳定性和容错性都是有对应的要求.实际开发项目中的代码与样例代码的区别,更多的是在代码的运行的稳定性.容错性.扩展性的比较.因为对于实现一个功能来说,实现功能的核心代码是一样的,可能只是在写法

  • ASP.NET Core异常和错误处理(8)

    在这一章,我们将讨论异常和错误处理.当 ASP.NET Core应用程序中发生错误时,您可以以各种不同的方式来处理.让我们来看看通过添加一个中间件来处理异常情况,这个中间件将帮助我们处理错误. 要模拟出错,让我们转到应用程序,运行,如果我们只是抛出异常的话,看看程序是如何运转转的. using Microsoft.AspNet.Builder; using Microsoft.AspNet.Hosting; using Microsoft.AspNet.Http; using Microsoft

  • 一些.NET对多线程异常处理技巧分享

    多线程环境 在我们的产品 SE 中,出现多线程的地方主要有两大类,一类是通过 ThreadPool 或 new Thread 主动发起多线程,另一类是 Socket 通讯回调. 多线程异常捕获 对于一般的异常处理来说,我们只要简单的将可能出错的语句包含在 try/catch 语句中即可.我也曾经简单的将该方法运用于多线程的异常捕获,结果并非如此,代码如下: 复制代码 代码如下: public static void Main() {     try     {         new Threa

随机推荐