微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

java – 代码覆盖最终阻止

我有以下代码构造:
try {
   //some code
}
catch(CustomException custExc) {
  //log
}
catch(CustomException2 custExc2) {
  //log
}
catch(Exception exc) {
  //log
}
finally {
  //some code
}

我写了单元测试:第一个是在不抛出异常的情况下覆盖的情况(仅执行块代码,最后阻止代码),另外3个是同时覆盖每个catch块的(执行try块,catch阻止并最终阻止).
问题是Eclipse emma插件显示我没有覆盖最后的块.任何想法为什么会发生?

解决方法

在Java字节码(至少从Java 1.6开始)之后,finally块没有特殊的构造,所以实际上重复了很多次.例如,考虑以下方法
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 举报,一经查实,本站将立刻删除。

相关推荐