很多帖子都分析过Try-Catch的机制,以及其对性能的影响。
但是并没有证据证明,Try-Catch过于损耗了系统的性能,尤其是在托管环境下。记得园子里有位网友使用StopWatch分析过Try-Catch在不同情况下,与无Try-Catch的代码相比,代码运行的时间指标,结果并没有很大差异。
下面我来结合IL分析一下Try-Catch吧。
● 机制分析
.Net 中基本的异常捕获与处理机制是由try…catch…finally块来完成的,它们分别完成了异常的监测、捕获与处理工作。一个try块可以对应零个或多个catch块,可以对应零个或一个finally块。不过没有catch的try似乎没有什么意义,如果try对应了多个catch,那么监测到异常后,CLR会自上而下搜索catch块的代码,并通过异常过滤器筛选对应的异常,如果没有找到,那么CLR将沿着调用堆栈,向更高层搜索匹配的异常,如果已到堆栈顶部依然没有找到对应的异常,就会抛出未处理的异常了,这时catch块中的代码并不会被执行。所以距离try最近的catch块将最先被遍历到。
如有以下代码:
try { Convert.ToInt32("Try"); } catch (FormatException ex1) {string CatchFormatException = "CatchFormatException"; } catch (NullReferenceException ex2) {
string CatchNullReferenceException = "CatchNullReferenceException"; }
finally { string Finally = "Finally"; }
对应IL如下:
.method private hidebysig instance void Form1_Load(object sender, class [mscorlib]System.EventArgs e) cil managed { // Code size 53 (0x35) .maxstack 1 .locals init ([0] class [mscorlib]System.FormatException ex1, [1] string CatchFormatException, [2] class [mscorlib]System.NullReferenceException ex2, [3] string CatchNullReferenceException, [4] string Finally) IL_0000: nop IL_0001: nop IL_0002: ldstr "Try" IL_0007: call int32 [mscorlib]System.Convert::ToInt32(string) IL_000c: pop IL_000d: nop IL_000e: leave.s IL_0026 IL_0010: stloc.0 IL_0011: nop IL_0012: ldstr "CatchFormatException" IL_0017: stloc.1 IL_0018: nop IL_0019: leave.s IL_0026 IL_001b: stloc.2 IL_001c: nop IL_001d: ldstr "CatchNullReferenceException" IL_0022: stloc.3 IL_0023: nop IL_0024: leave.s IL_0026 IL_0026: nop IL_0027: leave.s IL_0033 IL_0029: nop IL_002a: ldstr "Finally" IL_002f: stloc.s Finally IL_0031: nop IL_0032: endfinally IL_0033: nop IL_0034: ret IL_0035: // Exception count 3 .try IL_0001 to IL_0010 catch [mscorlib]System.FormatException handler IL_0010 to IL_001b .try IL_0001 to IL_0010 catch [mscorlib]System.NullReferenceException handler IL_001b to IL_0026 .try IL_0001 to IL_0029 finally handler IL_0029 to IL_0033 } // end of method Form1::Form1_Load
末尾的几行代码揭示出IL是怎样处理异常处理的。最后三行的每一个Item被称作Exception Handing Clause,EHC组成Exception Handing Table,EHT与正常代码之间由ret返回指令隔开。
可以看出,FormatException排列在EHT的第一位。
当代码成功执行或反之而返回后,CLR会遍历EHT:
1. 如果抛出异常, CLR会根据抛出异常的代码的“地址”找到对应的EHC(IL_0001 to IL_0010为检测代码的范围),这个例子中CLR将找到2条EHC,FormatException会最先被遍历到,且为适合的EHC。
2. 如果返回的代码地址在IL_0001 to IL_0029内,那么还会执行finally handler 即IL_0029 to IL_0033中的代码,不管是否因成功执行代码而返回。
事实上,catch与finally的遍历工作是分开进行的,如上文所言,CLR首先做的是遍历catch,当找到合适的catch块后,再遍历与之对应finally;而且这个过程会递归进行至少两次,因为编译器将C#的try…catch…finally翻译成IL中的两层嵌套。
当然如果没有找到对应的catch块,那么CLR会直接执行finally,然后立即中断所有线程。Finally块中的代码肯定会被执行,无论try是否检测到了异常。
改进建议
由上面的内容可以得出:
如果使用了“Try-Catch”,且捕获到了异常,CLR做的只不过是遍历Exception Handing Table中的Catch项;然后再次遍历Exception Handing Table中的Finally项,所用时间几乎都花费在遍历Exception Handing Table上;而如果没有捕获到异常,CLR只是遍历Exception Handing Table中的Finally项,所需时间微乎其微。
而“Try-Catch”遍历后的执行对应操作所用时间,则根据你的具体代码所定,“Try-Catch”引起的只是监控与触发,不应将这部分的代码时间也算“Try-Catch”的消耗。
所以,可以从性能和代码评审两方面考虑,一般建议有以下几点准则:
1.尽量给CLR一个明确的异常信息,不要使用Exception去过滤异常
2.尽量不要将try…catch写在循环中
3. try尽量少的代码,如果有必要可以使用多个catch块,并且将最有可能抛出的异常类型,书写在距离try最近的位置
4.不要只声明一个Exception对象,而不去处理它。这样做白白增加了Exception Handing Table的长度。
5.使用性能计数器实用工具的“CLR Exceptions”检测异常情况,并适当优化
6.使用成员的Try-Parse模式,如果抛出异常,那么用false代替它
结论,Try-Catch虽然会消费一点时间,但程序人员大可不必谈虎色变,通过上面的分析,与其说“Try-Catch”会损耗或影响性能,不如说“Try-Catch”与其他代码一样,只是性能的普通消费者,但出于代码书写评审方面的考虑,还是尽量关照一下“Try-Catch”吧。
我想知道如果我在一个while循环中使用一个或两个try catch,这是否影响性能? 下面的示例代码: 我的问题以前可能有人问过,但我也想知道是否比更好。 编辑: 我知道如果抛出,那么捕捉它需要一些时间,但是如果程序没有出现异常,情况如何:
在实际开发中,根据 try catch 语句的执行过程,try 语句块和 catch 语句块有可能不被完全执行,而有些处理代码则要求必须执行。例如,程序在 try 块里打开了一些物理资源(如数据库连接、网络连接和磁盘文件等),这些物理资源都必须显式回收。 Java的垃圾回收机制不会回收任何物理资源,垃圾回收机制只回收堆内存中对象所占用的内存。 所以为了确保一定能回收 try 块中打开的物理资源,异
我刚刚开始用java编写一个21点游戏。我试图让程序要求用户再次输入,如果他们输入的现金不是一个有效的整数。我看到许多带有catch的try语句示例,但没有一个是有效的。程序给出的错误InputMismatchException无法解析为类型。我遵循的一个线程就是这个,我有完全相同的代码,只是变量名不同。给你。Java输入不匹配异常 这是我的密码: 任何关于为什么我几乎精确的代码不起作用的帮助都将
问题内容: 在PHP 5中使用try-catch语句时,要考虑什么样的性能影响? 以前,我已经在网上阅读了一些有关该主题的古老的,看似矛盾的信息。我目前必须使用的许多框架都是在php 4上创建的,并且缺少php 5的许多优点。因此,我在使用try-catchs与php方面经验不足。 问题答案: 要考虑的一件事是,没有引发异常的try块的开销与实际引发和捕获异常的开销是一个不同的问题。 如果仅在失败
主要内容:多重catch语句在实际应用中,对于错误的处理是极其重要的,任何程序都很难做到百分百完美,程序中可能存在大量未知问题,所以程序开发时一定要对各种问题进行相应的处理,而 Java 提供的异常处理机制可以帮用户更好地解决这方面的问题。Java 的异常处理机制可以让程序具有极好的容错性,让程序更加健壮。 Java 的异常处理通过 5 个关键字来实现: try、 catch、 throw、 throws 和 finally
问题内容: 在内部使用try / catch或在内部使用try / catch的正确方法是什么? 与 问题答案: 在我看来,我的观点是: 以上是正确的方法。 因为,使用这种方法,如果与数据库的连接存在异常,则将其捕获在catch块内。但是使用第一种方法,则不会。