java – 代码覆盖最终阻止

前端之家收集整理的这篇文章主要介绍了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).

猜你在找的Java相关文章