目錄
Emit異常處理流程
顯示Exception對象的Message屬性
返回目錄
Emit異常處理流程
來看這種C#異常處理代碼:
{
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.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;
}
輸出:
finally塊
捕獲Exception
finally塊
如下代碼:
//+ 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;
}
新聞熱點
疑難解答