我在一个相关的任务中偶然发现了这个问题,虽然它很旧,但我认为它可能仍然对某人感兴趣。
方法编号
主要问题是intellij覆盖率和jacoco计算数字是否不同,哪种方式是正确的。简要回答:intellij覆盖率摘要使用开发人员直接提供的方法,而jacoco在字节码级别运行并显示在那里找到的方法数量。为了演示这一点,我创建了一个包含四种方法的简单类:
public class Exp {
private final LinkedList<Integer> vals = new LinkedList<>();
public void addVal(int v) {
vals.add(v);
}
public List<Integer> doubled() {
return vals.stream().map(x -> x*2).collect(Collectors.toList());
}
public List<Integer> evens() {
return vals.stream().filter(x -> x%2 == 0).collect(Collectors.toList());
}
public static void main(String[] args) {
Exp t = new Exp();
t.addVal(1);
t.addVal(2);
t.addVal(3);
System.out.println(t.doubled());
System.out.println(t.evens());
}
}
在 intellij 摘要中,右侧显示以下值:
因此,方法数等于示例代码中的方法数。Jacoco报告了七种方法,如报告中所示(与eclipse 2020-09中的Emma插件相同):
这是我们可以在字节码中找到的方法数量,例如通过使用javap反汇编器命令。在这里,我们看到两个 lambda 表达式作为类的方法实现,并且还插入了一个标准构造函数。
C:\_workspace\intellij\Tests\out\production\mod>javap -p Exp.class
Compiled from "Exp.java"
public class Exp {
private final java.util.LinkedList<java.lang.Integer> vals;
public Exp();
public void addVal(int);
public java.util.List<java.lang.Integer> doubled();
public java.util.List<java.lang.Integer> evens();
public static void main(java.lang.String[]);
private static boolean lambda$evens$1(java.lang.Integer);
private static java.lang.Integer lambda$doubled$0(java.lang.Integer);
}
让我有点困惑的是,intellij覆盖率报告(Run->Generate Corevage Report)显示了五种方法:
向代码中添加标准构造函数并重新生成报告会显示报告包含生成的标准构造函数,但不包含 lambda 表达式。似乎有一种中间计数方法。
至于intellij或jacoco是否正确的问题,我会说它们都是对的,这只是一个定义问题。
行号
在我的测试中,所有报告都显示一致的行号。在上面的示例中,报告了 13 行包含可执行代码的行。我对覆盖范围摘要中的 intellij 行计数的印象是,它不能一直正确刷新。可能需要进行干净的重建。