注解驱动: TestNG 使用注解来标识测试方法、测试类和配置方法,使得测试更具可读性。
并行执行: TestNG 支持多线程并行执行测试,可以加速测试套件的执行。
丰富的配置: 可以通过 XML 配置文件来定义测试套件的执行顺序、依赖关系等。
分组测试: 可以将测试方法分组,实现对特定测试分组的执行。
参数化测试: 可以使用 @Parameters 和@ DataProvider注解实现参数化测试
依赖管理: 可以通过 dependsOnMethods 或 dependsOnGroups 设置测试方法之间的依赖关系。
断言灵活: TestNG 提供了丰富的断言方法,用于验证测试结果。
@Test:用于标识测试方法,执行单元测试。
@BeforeSuite:在测试套件之前执行。
@AfterSuite:在测试套件之后执行。
@BeforeTest:在测试类之前执行。
@AfterTest:在测试类之后执行。
@BeforeClass:在测试类中的所有测试方法之前执行。
@AfterClass:在测试类中的所有测试方法之后执行。
@BeforeMethod:在每个测试方法之前执行。
@AfterMethod:在每个测试方法之后执行。
@DataProvider:用于提供测试数据。
@Parameters:用于参数化测试。
@dependsOnMethods:设置测试方法的依赖关系。
package AI.Test.testNG;
import org.testng.annotations.*;
public class TestNGCheck {
@BeforeSuite
public void beforeSuite(){
System.out.println("beforeSuite");
}
@BeforeTest
public void beforeTest() {
System.out.println("Before Test");
}
@BeforeClass
public void beforeClass() {
System.out.println("Before Class");
}
@BeforeMethod
public void beforeMethod() {
System.out.println("Before Method");
}
@Test(groups="Performance")
public void testMethod1() {
System.out.println("Test Method 1");
}
@AfterMethod
public void afterMethod() {
System.out.println("After Method");
}
@AfterClass
public void afterClass() {
System.out.println("After Class");
}
@AfterTest
public void afterTest() {
System.out.println("After Test");
}
@AfterSuite
public void afterSuite() {
System.out.println("After Suite");
}
}
1. 常用的:
groups: 用于将测试方法分组,可以在 XML 配置中选择性地执行指定组的测试。
timeOut: 指定方法的超时时间,单位为毫秒。
alwaysRun: 如果设置为 true,则无论依赖的方法是否失败,当前方法都会执行。
dependsOnMethods: 指定测试方法的依赖关系,被依赖的方法会在当前方法之前执行。
enabled: 控制测试方法是否启用,设置为 false 则禁用该方法。
description: 提供关于测试方法的简要描述。
invocationCount: 指定测试方法被调用的次数。
2. 数据驱动,需要结合@DataProvider注解对应获取参数的方法:
dataProvider: 指定使用哪个数据提供者来提供测试数据。
dataProviderClass: 指定数据提供者所在的类。
3.并发:
invocationCount: 指定测试方法被调用的次数。
threadPoolSize: 指定线程池的大小,用于并行执行测试方法。
dataProviderThreadCount: 指定数据提供者线程的数量。
举例:
@Test(groups="Performance")
public void testMethod1() {
System.out.println("Test Method 1");
}
@Test(priority=1)
public void testMethod2() {
System.out.println("Test Method 2");
}
@Test(dependsOnMethods = "testMethod4", alwaysRun = false)
public void testMethod3(){
System.out.println("Test Method 3");
}
@Test(timeOut = 3000)
public void testMethod4() throws InterruptedException {
Thread.sleep(4000);
System.out.println("Test Method 4");
}
@Test(enabled = false, description = "用于测试enable注解")
public void testMethod5() {
System.out.println("Test Method 5");
}
以Yaml文件为例
创建一个Yaml文件testdata.yaml
testdata:
- parameters: [2, 3, 5]
- parameters: [10, 20, 30]
获取测试数据及使用
import org.testng.annotations.DataProvider;
import org.testng.annotations.Test;
import org.yaml.snakeyaml.Yaml;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
public class ParameterizedTest {
@DataProvider(name = "testData")
public Object[][] getYamlData() {
List
5. 执行顺序
TestNG 默认情况下会按照测试方法的名称升序执行。
可以通过设置 priority 属性来指定执行顺序。
也可以使用dependsOnMethods
6. testNG断言
assertEquals(expected, actual):
验证两个值是否相等。如果不相等,会抛出 AssertionError。
assertNotEquals(expected, actual):
验证两个值是否不相等。如果相等,会抛出 AssertionError。
assertTrue(condition):
验证给定的条件是否为真。如果条件为假,会抛出 AssertionError。
assertFalse(condition):
验证给定的条件是否为假。如果条件为真,会抛出 AssertionError。
assertNull(object):
验证给定的对象是否为 null。如果对象不为 null,会抛出 AssertionError。
assertNotNull(object):
验证给定的对象是否不为 null。如果对象为 null,会抛出 AssertionError。
assertSame(expected, actual):
验证两个引用是否指向同一个对象。如果不指向同一个对象,会抛出 AssertionError。
assertNotSame(expected, actual):
验证两个引用是否指向不同的对象。如果指向同一个对象,会抛出 AssertionError。
现代化的设计: JUnit 5 的架构更加模块化和现代化,采用了模块化体系结构,使得集成和扩展更加容易。它引入了 Jupiter(新的测试引擎)和 Vintage(支持 JUnit 4)两个模块,具有更好的灵活性。
Lambda 表达式支持: JUnit 5 充分利用了 Java 8 的 Lambda 表达式特性,使测试代码更加简洁。TestNG 也支持 Lambda 表达式,但 JUnit 5 的设计更加贴合现代 Java 特性。
参数化测试: JUnit 5 提供了内置的参数化测试功能,使用 @ParameterizedTest 注解,可以轻松地对测试方法使用不同的参数运行。
动态测试: JUnit 5 引入了动态测试,允许在运行时生成和执行测试。使用 @TestFactory 注解,可以动态生成测试方法,以适应更加灵活的测试需求。
扩展机制: JUnit 5 的扩展机制更加强大和灵活,使用 @ExtendWith 注解,可以轻松应用自定义扩展,甚至可以自定义扩展来修改测试运行时的行为。
条件测试: JUnit 5 允许根据条件决定是否执行测试方法,使用 @EnabledOnOs、@DisabledIf 等注解,可以根据操作系统、环境变量等条件来控制测试执行。
并行执行: TestNG 在并行执行方面有很强的功能,但 JUnit 5 也在逐步增强并行执行的支持,对于简单的并行需求,JUnit 5 也可以胜任。
灵活性: JUnit 5 允许更多的自定义配置,使得测试执行的控制更加灵活,可以根据不同的项目需求进行调整。
@Test: 用于标记测试方法。
@DisplayName: 为测试方法或测试类指定一个可读的名称。
@BeforeEach: 在每个测试方法之前执行的方法。
@AfterEach: 在每个测试方法之后执行的方法。
@BeforeAll: 在所有测试方法之前执行的方法,必须是静态方法。
@AfterAll: 在所有测试方法之后执行的方法,必须是静态方法。
@Disabled: 标记测试方法或测试类为禁用状态。
@ParameterizedTest: 用于参数化测试的注解。
@RepeatedTest: 用于指定重复执行测试方法的次数。
@Timeout: 用于设置测试方法执行的最大时间。时间是s
@Tag: 为测试方法添加标签,用于分组和过滤测试。
@Nested: 用于嵌套测试类。
@TestFactory: 用于动态测试,返回动态生成的测试方法。
package AI.Test.testNG;
import org.junit.jupiter.api.*;
public class Junit5TestCheck {
@BeforeAll
static void beforeAll() throws InterruptedException {
System.out.println("Before All");
Thread.sleep(3000);
}
@BeforeEach
void beforeEach(){
System.out.println("BeforeEach");
}
@Test
@DisplayName("Junit5TestCheck test1")
void test1(){
System.out.println("test1");
}
@Test
@RepeatedTest(2)
@DisplayName("Junit5TestCheck test2")
void test2(){
System.out.println("test2");
}
@Test
@Disabled
@DisplayName("Junit5TestCheck test3")
void test3(){
System.out.println("test3");
}
@Test
@Timeout(3)
void test4() throws InterruptedException {
Thread.sleep(4);
System.out.println("test4");
}
@AfterEach
void afterEach(){
System.out.println("AfterEach");
}
@AfterAll
static void afterAll(){
System.out.println("After All");
}
}
TestFactory其实就是参数化执行同一条用例
通常情况下,我们会使用 @ParameterizedTest
注解来参数化执行不同的测试用例,每次测试会使用不同的参数进行运行。而 @TestFactory
则是一种更为动态和灵活的方式,它允许你在运行时生成测试用例。
import org.junit.jupiter.api.TestFactory;
import org.junit.jupiter.api.DynamicTest;
import org.junit.jupiter.api.Test;
import java.util.stream.Stream;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.DynamicTest.dynamicTest;
public class DynamicTestExample {
@TestFactory
Stream dynamicTestFactory() {
return Stream.of(
dynamicTest("Test 1", () -> assertEquals(2, add(1, 1))),
dynamicTest("Test 2", () -> assertEquals(4, add(2, 2))),
dynamicTest("Test 3", () -> assertEquals(6, add(3, 3)))
);
}
int add(int a, int b) {
return a + b;
}
}
4. ParameterizedTest
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.CsvSource;
import static org.junit.jupiter.api.Assertions.assertEquals;
public class ParameterizedTestExample {
@ParameterizedTest
@CsvSource({"1, 2, 3", "0, 0, 0", "-1, -2, -3"})
void testAddition(int a, int b, int result) {
assertEquals(result, add(a, b));
}
int add(int a, int b) {
return a + b;
}
}