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

时间:2021-05-21

目录

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!

声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。

相关文章