使用JUnitCore运行参数化测试

时间:2016-02-11 23:06:46

标签: java junit junit-runner

是否可以使用JUnitCore API运行参数化测试类?

我正在测试一个名为 Fibonacci 的类,一个名为 TestFibonacci 的参数化测试类,以及一个简单的Java类( JUnitParameterized )使用JUnitCore API执行 TestFibonacci 类。如果我使用JUnit插件或命令行执行 TestFibonacci ,它会通过。但是,当我使用 JUnitParameterized 类执行它时,它会失败。

待测班级

public class Fibonacci {

  public static int compute(int n) {
    if (n <= 1) {
      return n;
    }
    return compute(n-1) + compute(n-2);
  }
}

测试类

import static org.junit.Assert.assertEquals;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import org.junit.runners.Parameterized.Parameters;

import java.util.Arrays;

@RunWith(Parameterized.class)
public class TestFibonacci {

  @Parameters(name = "{index}: fib({0})={1}")
  public static Iterable<Object[]> data() {
    return Arrays.asList(
        new Object[][] { { 0, 0 }, { 1, 1 }, { 2, 1 }, { 3, 2 }, { 4, 3 }, { 5, 5 }, { 6, 8 } });
  }

  private int input;
  private int expected;

  public TestFibonacci(int input, int expected) {
    this.input = input;
    this.expected = expected;
  }

  @Test
  public void test() {
    assertEquals(expected, Fibonacci.compute(input));
  }
}

Java程序

import org.junit.runner.JUnitCore;
import org.junit.runner.Request;
import org.junit.runner.Result;

public class JUnitParameterized {

  public static void main(String[] args) throws ClassNotFoundException {

    Class<?> testClass = JUnitParameterized.class.getClassLoader().loadClass(TestFibonacci.class.getCanonicalName());

    Result result = (new JUnitCore()).run(Request.method(testClass, "test"));
    System.out.println("Number of tests run: " + result.getRunCount());
    System.out.println("The number of tests that failed during the run: " + result.getFailureCount());
    System.out.println("The number of milliseconds it took to run the entire suite to run: " + result.getRunTime());
    System.out.println("" + (result.wasSuccessful() == true ? "Passed :)" : "Failed :("));
  }
}

1 个答案:

答案 0 :(得分:3)

当JUnit测试类使用@Parameterized注释时,测试方法名称作为描述,使用大括号,冒号和您在“c {”中提供的替换name中的数字进行了丰富。 @Parameters注释。

在您的情况下,要执行单个测试,即设置一个参数,您将

Result result = (new JUnitCore()).run(Request.method(TestFibonacci.class, "test[6: fib(6)=8]"));

在这种情况下,提供第6个参数集(从零开始) 请注意,您必须将您在方法名称中提供的数字与您声明为参数的数字完全匹配。序列号也很重要。因为这样做不起作用:

 "test[3: fib(6)=8]" (wrong sequence .. <6, 8> pair is 6th, not 3rd)
 "test[6: fib(50)=100]" (the pair <50, 100> is not declared in parameters)

为了避免对@Parameters(name=?)值进行不必要的解析,我建议只使用name声明参数:

@Parameters // no name=?
public static Iterable<Object[]> data() {
    return Arrays.asList(
         new Object[][] { { 0, 0 }, { 1, 1 }, { 2, 1 }, { 3, 2 }, { 4, 3 }, { 5, 5 }, { 6, 8 } });
}

然后测试方法描述只是test[6]

Result result = (new JUnitCore()).run(Request.method(TestFibonacci.class, "test[6]"));

要使用所有参数运行单个测试方法,我建议在一个循环中执行它(并可能聚合结果):

int parametersCount = Request.aClass(TestFibonacci.class).getRunner().getDescription().getChildren().size();
for (int i = 0; i < parametersCount; ++i) {
    Result result = (new JUnitCore()).run(Request.method(testClass, "testFloat[" + i + "]"));
    System.out.println("Result " + result.wasSuccessful());
}