Net有道

紫冠道人的求道历程

导航

<2005年1月>
2627282930311
2345678
9101112131415
16171819202122
23242526272829
303112345

与我联系

搜索

 

常用链接

留言簿(21)

我参与的团队

随笔分类

随笔档案

文章分类

文章档案

Program

程序之外的生活

我的阅读BLOG

积分与排名

最新评论

阅读排行榜

评论排行榜

在多重Catch的情况下得到异常的完整信息

 

在方法多层调用的时候,每一层都有相应的catch处理后重新throw的情况下,刚开始在最外层无法得到产生异常的完整信息。

最初的代码样子如下:

static void Main(string[] args)

         {

              try{

                   Method3();

              }catch(Exception e) {

                   Console.WriteLine(e.ToString());

              }

 

              Console.Read();

         }

 

         static void Method1(int a,int b) {

              try{

                   int c = a / b;

              }catch(DivideByZeroException){

                   throw new CustomException("Parameter b can't be 0");

              }

         }

 

         static void Method2() {

              try{

                   。。。

                   Method1(1,0);

              }catch(CustomException e){

                   。。。

                   throw e;

              }

         }

 

         static void Method3() {

              try{

                   Method2();

                   。。。

              }catch(CustomException e) {

                   。。。

                   throw e;

              }

     }

(这只是示例,实际调用的时候当然不会太有同一个类里调用这么深的,有可能会在好几个类之间调用。CustomException是一个自定义的异常)。

输出:

CustomException.CustomException: Parameter b can't be 0

   at CustomException.Class1.Method3() in 。。。\class1.cs:line 58

   at CustomException.Class1.Main(String[] args) in 。。。\class1.cs:line 19

 

Method1里出的异常,可显示出来的信息只到了Method3。在代码错综复杂的情况,出的这种信息有点让人不知所措。

 

看《.NET框架程序设计(修改版)》一书得知,原来小改一下就好多了。把所有throw e;的地方改为throw;就可以得到下面的输出:

CustomException.CustomException: Parameter b can't be 0

   at CustomException.Class1.Method1(Int32 a, Int32 b) in。。。\class1.cs:line 41

   at CustomException.Class1.Method2() in 。。。\class1.cs:line 50

   at CustomException.Class1.Method3() in。。。\class1.cs:line 58

   at CustomException.Class1.Main(String[] args) in 。。。\class1.cs:line 19

原来在throw e;这样写的时候,CLR会重新设置异常的起始点,所以就把之前的信息给丢失了。

 

这明显比第一次好多了。但很多时候,我们不会把异常经相应的处理后原样抛出,有时会转化为另一个异常后再抛出,这种情况下又是与刚开始时一样的,会丢失信息。这种丢失虽说是预期的,本就是要对外表现为那个样子的,但在自己开发调试的时候这种情况很烦,总是无法第一时间找到异常发生的比较确切的地方,只是缩小一下范围而已。

这个时候就要用那个带InnerException参数的构造器了(书上说在FCL中很多异常类型没有提供相应的构造器,但我碰到几个怎么都是有的;而且我们现在多是一些自定义异常为多),然后用ToString方法得到完整的信息。

最后:

         static void Main(string[] args)

         {

              try{

                   Method3();

              }catch(Exception e) {

                   Console.WriteLine(e.ToString());

              }

 

              Console.Read();

         }

 

         static void Method1(int a,int b) {

              try{

                   int c = a / b;

              }catch(DivideByZeroException e){

                   throw new CustomException("Parameter b can't be 0",e);

              }

         }

 

         static void Method2() {

              try{

                   。。。

                   Method1(1,0);

              }catch(CustomException){

                   throw;

              }

         }

 

         static void Method3() {

              try{

                   Method2();

                   。。。

              }catch(Exception e) {

                   throw new CustomException("Invoke Method3 Error",e);

              }

     }

得到:

CustomException.CustomException: Invoke Method3 Error ---> CustomException.Custo

mException: Parameter b can't be 0 ---> System.DivideByZeroException: 试图除以零

   at CustomException.Class1.Method1(Int32 a, Int32 b) in。。。\class1.cs:line 36

   --- 内部异常堆栈跟踪的结尾 ---

   at CustomException.Class1.Method1(Int32 a, Int32 b) in。。。\class1.cs:line 44

   at CustomException.Class1.Method2() in。。。\class1.cs:line 52

   at CustomException.Class1.Method3() in。。。\class1.cs:line 58

   --- 内部异常堆栈跟踪的结尾 ---

   at CustomException.Class1.Method3() in。。。\class1.cs:line 60

   at CustomException.Class1.Main(String[] args) in。。。\class1.cs:line 19

 

这找起错误来就方便多了。


posted on 2005-01-28 10:28 lichdr 阅读(1313) 评论(6)  编辑 收藏 所属分类: CLR

评论

#1楼  2005-01-28 13:26 idior [未注册用户]

good 实用的小技巧   回复  引用    

#2楼  2005-01-28 13:36 kc [未注册用户]

这样很麻烦,如果是为了捕捉最原始的异常地方和信息
直接不用catch就是啦   回复  引用    

#3楼  2005-01-28 15:16 tornado      

如kc所说,编程时如果想得到完整的原始异常,在底层或中间层的一些方法什么的就没必要异常提示,直接throw出来(也就是try{}catch{throw;})
只需在最外层才加上给用户明确的异常提示这样也可以。   回复  引用  查看    

#4楼 [楼主] 2005-01-28 15:38 lichdr      

中间不放任何的catch,到最外面catch一下当然就是可以的。

但文中所讨论的情形本就是矛盾的情况。中途捕获异常一方面是要额外的处理(这个还好说),处理完后直接throw;就是了。
但有些还是要转化为另一种异常向外表现,但又不想迷惑了自己。

在实际的开发中在最外层是不会写 Console.WriteLine(e.ToString());这样的东西的。这个不会展示给用户。只是自己用用。
  回复  引用  查看    

#5楼  2005-01-28 21:52 kc [未注册用户]

有些还是要转化为另一种异常向外表现,但又不想迷惑了自己。
既然是自己写的异常表现,怎么又会迷惑自己   回复  引用    

#6楼  2005-01-28 22:20 小残 [未注册用户]

自己写的异常表现,可能只是简单地提示用户出现了异常,并没有显示详细的异常信息,一来异常的具体信息对用户而言不太友善,二来也可能会暴露一些敏感的信息。
没有了详细的异常信息,当然可能被迷惑。   回复  引用    


标题  
姓名  
主页
Email (博主才能看到) 
验证码 *  看不清,换一张 [登录][注册]
内容(请不要发表任何与政治相关的内容)  
  登录  使用高级评论  新用户注册  返回页首  恢复上次提交      


相关链接: