Java 新特性在测试自动化框架中的应用

2025-05-11   出处: Mediam  作/译者:Govinda Solanki/溜的一比

Java 新特性在测试自动化框架中的应用

Java 持续发展,不断推出的新特性可以显著增强您的测试自动化框架。以下是最有影响力的近期新增功能,您可以利用它们来优化您的测试项目。

字符串模板(Java 21 预览版,Java 24)

Java 21 和 24 中的字符串模板显著改进了我们在测试自动化代码中创建动态字符串的方式。

在 Java 中,我们传统上使用字符串拼接或 String.format()​ 来创建包含变量内容的字符串。这两种方法都有缺点:

  • 字符串拼接在变量较多时会变得混乱。
  • String.format()​ 需要匹配格式说明符(如 %s​、%d​)与变量的顺序。

两者在复杂字符串中都容易出错且难以阅读。

字符串模板允许您直接在字符串中清晰可读地嵌入表达式。

// 使用新的字符串模板功能
// STR. 前缀表示这是一个模板处理器
// 使用花括号 \{expression} 插入值
// 花括号内可以放置任何有效的 Java 表达式

String name = "Test";
int age = 25;
String message = STR."Hello, my name is \{name} and I am \{age} years old.";

@Test
void testUserRegistration() {
    // 测试数据
    String username = "testUser123";
    String email = "test@example.com";
    String password = "SecureP@ss";

    // 使用字符串模板创建 JSON 负载
    String requestBody = STR."""
    {
      "username": "\{username}",
      "email": "\{email}",
      "password": "\{password}",
      "createDate": "\{LocalDate.now()}"
    }
    """;

    // 在测试中使用此请求体
    Response response = apiClient.post("/users", requestBody);

    // 断言
    assertEquals(201, response.getStatusCode());
}

模式匹配(Java 16+)

模式匹配用于 instanceof​,彻底改变了我们在测试代码中处理类型检查和类型转换的方式。此功能减少了测试断言中的样板代码,使条件验证更加清晰。

// 旧方法
if (testResult instanceof SuccessResult) {
    SuccessResult success = (SuccessResult) testResult;
    verifySuccessData(success.getData());
}

// 新模式匹配方法
if (testResult instanceof SuccessResult success) {
    verifySuccessData(success.getData());
}

记录类(Java 16+)

记录类非常适合在测试代码中创建简单的、不可变的数据载体。记录类消除了为测试数据对象编写构造函数、getter、equals()​、hashCode()​ 和 toString()​ 方法的需要。

// 简洁定义测试数据结构
record TestUser(String username, String email, boolean isActive) {}
record ExpectedResponse(int statusCode, String body, Map<String, String> headers) {}

@Test
void verifyUserRegistration() {
    TestUser user = new TestUser("testUser", "test@example.com", true);
    ExpectedResponse expected = new ExpectedResponse(200, "Success", Map.of("Content-Type", "application/json"));

    // 在测试中使用这些不可变对象
    ResponseEntity response = userService.register(user);
    assertEquals(expected.statusCode(), response.getStatusCode().value());
}

文本块(Java 15+)

文本块使处理多行字符串变得更加容易,特别适用于 JSON/XML 测试数据和 SQL 查询。

String jsonTestData = """
    {
      "username": "testUser",
      "password": "securePass123",
      "roles": ["user", "admin"],
      "settings": {
        "notifications": true,
        "theme": "dark"
      }
    }
    """;

Stream.toList()(Java 16+)

这是一个小但重要的改进,简化了流结果的收集。

// 旧方法
List<String> errorMessages = testResults.stream()
    .filter(result -> !result.isPassed())
    .map(TestResult::getMessage)
    .collect(Collectors.toList());

// 新方法
List<String> errorMessages = testResults.stream()
    .filter(result -> !result.isPassed())
    .map(TestResult::getMessage)
    .toList();

Stream.mapMulti()(Java 16+)

此方法适用于展平复杂的测试数据结构。

List<TestScenario> flattenedTestCases = testSuites.stream()
    .mapMulti((suite, consumer) -> {
        suite.getTestCases().forEach(consumer);
    })
    .toList();

更精确的空指针异常(Java 14+)

此功能提供了关于链中哪个变量为 null 的精确信息。

// 当此代码因 NPE 失败时,异常消息会明确指出哪个部分为 null
user.getAddress().getCity().equals("New York");

// 您可以立即看到 getCity() 返回了 null,这直接指向问题的根源。
java.lang.NullPointerException: Cannot invoke "com.example.City.equals(Object)" because the return value of "com.example.Address.getCity()" is null
    at com.example.YourClass.yourMethod(YourClass.java:42)

Switch 表达式增强(Java 14+)

Switch 表达式使测试逻辑更加简洁。

TestResult evaluateResponse(HttpStatus status) {
    return switch(status) {
        case OK, CREATED, ACCEPTED -> new TestResult(true, "Success");
        case NOT_FOUND -> new TestResult(false, "Resource not found");
        case UNAUTHORIZED, FORBIDDEN -> new TestResult(false, "Authentication error");
        default -> new TestResult(false, "Unexpected status: " + status);
    };
}

字符串增强

String 类添加了几个有用的方法。

// isBlank() - Java 11
assertTrue(response.getBody().isBlank());

// strip() 方法 - Java 11
assertEquals("expected", response.getHeader("X-Value").strip());

// lines() - Java 11
List<String> logLines = testOutput.lines().toList();

// transform() - Java 12
String normalizedOutput = rawTestOutput.transform(String::toLowerCase)
                                      .transform(s -> s.replaceAll("\\s+", " "));

// formatted() - Java 15(替代 String.format)
String testMessage = "Expected %s but got %s".formatted(expected, actual);

// indent() - Java 12
String prettyJson = rawJson.indent(2);

增强的随机数生成(Java 17+)

伪随机数生成器得到了增强,适用于生成随机测试数据。

java复制

// 创建一个新的随机数生成器
RandomGenerator generator = RandomGenerator.of("L64X128MixRandom");

// 生成随机测试数据
int randomAge = generator.nextInt(18, 99);
double randomSalary = generator.nextDouble(30000, 150000);

确保您的项目配置为使用适当的 Java 版本(建议使用 Java 17+ 以支持大多数功能)。


声明:本文为本站编辑转载,文章版权归原作者所有。文章内容为作者个人观点,本站只提供转载参考(依行业惯例严格标明出处和作译者),目的在于传递更多专业信息,普惠测试相关从业者,开源分享,推动行业交流和进步。 如涉及作品内容、版权和其它问题,请原作者及时与本站联系(QQ:1017718740),我们将第一时间进行处理。本站拥有对此声明的最终解释权!欢迎大家通过新浪微博(@测试窝)或微信公众号(测试窝)关注我们,与我们的编辑和其他窝友交流。
15° /158 人阅读/0 条评论 发表评论

登录 后发表评论
最新文章