为什么 Intellij 代码覆盖率和 jacoco 代码覆盖率显示不同的百分比?

我在intellij中有一个gradle项目(java)。我右键单击intellij中的项目并运行,并在右侧创建了一些测试报告。在右手边,我有这样的数字Run Tests in projectName with coverage

| Class, %   | Method, %   | Line, %
--------------------------------------
80%(80/100)  50%(100/200)  30%(30/100)

注意:以上数字仅举例说明。这些都不是真的。

现在我去了命令行并运行,它给出了一个方法和行,但是.为什么在这种情况下存在差异?gradlew jacocoTestReportdifferent set of numbersClass numbers were same

有没有办法从命令行运行 intellij 的代码覆盖率,而不是右键单击?

我只是想知道Intellij是否使用不同的方式来计算这些数字,而不是jacoco。但即使在这种情况下,我的假设是只有一种方法可以计算东西对吧?或者intellij或jacoco不计算具有龙目岛注释等的类,从而减少了最终计数中的方法(getters和setters)的数量?


答案 1

我在一个相关的任务中偶然发现了这个问题,虽然它很旧,但我认为它可能仍然对某人感兴趣。

方法编号

主要问题是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 摘要中,右侧显示以下值:

intellij coverage summary

因此,方法数等于示例代码中的方法数。Jacoco报告了七种方法,如报告中所示(与eclipse 2020-09中的Emma插件相同):jacoco test report

这是我们可以在字节码中找到的方法数量,例如通过使用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)显示了五种方法:

intellij coverage report

向代码中添加标准构造函数并重新生成报告会显示报告包含生成的标准构造函数,但不包含 lambda 表达式。似乎有一种中间计数方法。

至于intellij或jacoco是否正确的问题,我会说它们都是对的,这只是一个定义问题。

行号

在我的测试中,所有报告都显示一致的行号。在上面的示例中,报告了 13 行包含可执行代码的行。我对覆盖范围摘要中的 intellij 行计数的印象是,它不能一直正确刷新。可能需要进行干净的重建。


答案 2

我注意到我使用的@Test注释来自Jacoco没有拾取的注释。更改导入以修复我的承保范围问题。org.junit.jupiter.api.Testimport org.junit.Test


推荐