我目前正在做我的第一个 Java 项目,并希望对其进行完全的 TDD。我正在使用 JUnit 编写测试。显然 JUnit 不提供对数据提供者的支持,这使得用 20 个不同版本的参数测试同一个方法相当烦人。支持数据提供者的最流行/标准的 Java 测试工具是什么?我遇到了TestNG,但不知道它有多受欢迎,或者它与替代品相比如何。
如果有一种方法可以让这种行为是使用 JUnit 的好方法,那么这也可能有效。
我在我们公司的同事为 JUnit 编写了一个免费的 TestNG 风格的 DataProvider,你可以在 github (https://github.com/TNG/junit-dataprovider) 上找到它。
我们在非常大的项目中使用它,它对我们来说效果很好。它比 JUnit 有一些优势,Parameterized
因为它将减少单独类的开销,并且您也可以执行单个测试。
一个例子看起来像这样
@DataProvider
public static Object[][] provideStringAndExpectedLength() {
return new Object[][] {
{ "Hello World", 11 },
{ "Foo", 3 }
};
}
@Test
@UseDataProvider( "provideStringAndExpectedLength" )
public void testCalculateLength( String input, int expectedLength ) {
assertThat( calculateLength( input ) ).isEqualTo( expectedLength );
}
编辑:从 v1.7 开始,它还支持提供数据(字符串、列表)的其他方式,并且可以内联提供程序,因此不一定需要单独的方法。
可以在 github 的手册页上找到完整的工作示例。它还具有更多功能,例如在实用程序类中收集提供程序并从其他类访问它们等。手册页非常详细,我相信您会在那里找到任何问题的答案。
JUnit 4 具有参数化测试,它与 php 数据提供者做同样的事情
@RunWith(Parameterized.class)
public class MyTest{
@Parameters
public static Collection<Object[]> data() {
/*create and return a Collection
of Objects arrays here.
Each element in each array is
a parameter to your constructor.
*/
}
private int a,b,c;
public MyTest(int a, int b, int c) {
this.a= a;
this.b = b;
this.c = c;
}
@Test
public void test() {
//do your test with a,b
}
@Test
public void testC(){
//you can have multiple tests
//which all will run
//...test c
}
}
根据您对灵活性和可读性的需求,您可以选择Parameterized
- junit 的内置选项,由 dkatzel 描述。其他选项是zohhak等外部库提供的外部 junit运行器,您可以这样做:
@TestWith({
"clerk, 45'000 USD, GOLD",
"supervisor, 60'000 GBP, PLATINUM"
})
public void canAcceptDebit(Employee employee, Money money, ClientType clientType) {
assertTrue( employee.canAcceptDebit(money, clientType) );
}
或功能稍有不同的junitParams。只选择最适合你的
您可以使用 JUnit 5 的 ParameterizedTest。这是来自https://www.petrikainulainen.net/programming/testing/junit-5-tutorial-writing-parameterized-tests/的示例:
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.Arguments;
import java.util.stream.Stream;
import static org.junit.jupiter.api.Assertions.assertEquals;
@DisplayName("Should pass the method parameters provided by the sumProvider() method")
class MethodSourceExampleTest {
@DisplayName("Should calculate the correct sum")
@ParameterizedTest(name = "{index} => a={0}, b={1}, sum={2}")
@MethodSource("sumProvider")
void sum(int a, int b, int sum) {
assertEquals(sum, a + b);
}
private static Stream<Arguments> sumProvider() {
return Stream.of(
Arguments.of(1, 1, 2),
Arguments.of(2, 3, 5)
);
}
}
可以从注释、方法甚至 CSV 文件中加载测试参数。
这是另一种选择。您不必使用 Google Guava,这只是我的实现。
这使用与@Parameters
@dkatzel 的答案相同,但不是类接受参数,而是@Parameters
注释继续特定的测试方法,因此您可以选择哪些方法使用该组参数。
import java.util.Collection;
import com.google.common.collect.ImmutableList;
import junitparams.JUnitParamsRunner;
import junitparams.Parameters;
import org.junit.Assert;
import org.junit.Test;
import org.junit.runner.RunWith;
@RunWith(JUnitParamsRunner.class)
public class FrobTester {
@SuppressWarnings("unused")
private Collection validfrobAndGorpValues() {
return ImmutableList.of(
new Object[] {"frob28953", 28953},
new Object[] {"oldfrob-189-255", 1890255}
);
}
@Test
@Parameters(method = "validfrobAndGorpValues")
public void whenGivenFrobString_thenGorpIsCorrect(
String frobString,
int expectedGorpValue
) {
Frob frob = new Frob(frobString);
Assert.assertEquals(expectedGorpValue, frob.getGorpValue());
}
}