try { //some code } catch(CustomException custExc) { //log } catch(CustomException2 custExc2) { //log } catch(Exception exc) { //log } finally { //some code }
我写了单元测试:第一个是在不抛出异常的情况下覆盖的情况(仅执行块代码,最后阻止代码),另外3个是同时覆盖每个catch块的(执行try块,catch阻止并最终阻止).
问题是Eclipse emma插件显示我没有覆盖最后的块.任何想法为什么会发生?
解决方法
public static void main(String[] args) { try { System.out.println("In try"); if(args.length > 0) return; System.out.println("No args"); } catch(RuntimeException ex) { System.out.println("In catch"); } finally { System.out.println("In finally"); } }
此代码有效地编译成如下所示:
public static void main(String[] args) { try { System.out.println("In try"); if(args.length > 0) { System.out.println("In finally"); return; } System.out.println("No args"); } catch(RuntimeException ex) { System.out.println("In catch"); System.out.println("In finally"); } catch(<any exception> t) { System.out.println("In finally"); throw t; } System.out.println("In finally"); }
这不是一个完全等效的代码,因为如果在System.out.println(“In finally”)期间发生新的异常; (返回之前),那么它不会被捕获.然而,它显示了这个粗略的想法,finally块在这里重复了四次.如果您有几种方法可以退出尝试块,特别是如果您有嵌套的try-finally块,则可以复制更多次.另请注意< any exception>特别追加.即使你明确地写了catch(Throwable t),它也会出现在字节码中.
由于像emma或JaCoCo这样的代码覆盖工具在字节码级别上工作,他们并不知道这四个“In finally”printlns在源代码中实际上是一样的语句.可以执行字节码分析,并且相当鲁棒地确定字节码的哪个部分对应于单个源最终阻止(我实际上写了这样的分析器一次),但这不是很容易的问题,并且具有一些不平凡的注意事项.您还应该考虑到不同的编译器(例如,javac和ecj)会产生一些不同的finally块布局.所以似乎这个工作没有在流行的覆盖工具中完成,他们只是将不同的最终块拷贝视为不同的代码.
在你特定的情况下,@bobbel是正确的:你没有测试未捕获的异常情况(< any exception> catch).
原文地址:https://www.jb51.cc/java/125889.html
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。