SpringBootTDD工作流Skill springboot-tdd

本技能为Spring Boot后端开发提供完整的测试驱动开发(TDD)工作流指南。它详细介绍了如何使用JUnit 5、Mockito、MockMvc、Testcontainers和JaCoCo等工具,从单元测试、Web层测试到集成测试和持久层测试,确保代码质量并达到80%以上的测试覆盖率。适用于开发者在添加新功能、修复Bug或重构代码时,遵循先写测试再实现代码的TDD最佳实践,提升软件的可维护性和可靠性。关键词:Spring Boot, TDD, 测试驱动开发, JUnit, Mockito, 集成测试, 代码覆盖率, Java后端测试。

测试 0 次安装 24 次浏览 更新于 2/27/2026

name: springboot-tdd description: 使用JUnit 5、Mockito、MockMvc、Testcontainers和JaCoCo为Spring Boot进行测试驱动开发。适用于添加新功能、修复错误或重构代码时。

Spring Boot TDD 工作流

针对Spring Boot服务的TDD指南,目标达到80%+的覆盖率(单元测试 + 集成测试)。

何时使用

  • 新增功能或端点
  • 修复错误或重构代码
  • 添加数据访问逻辑或安全规则

工作流程

  1. 先写测试(它们应该会失败)
  2. 实现最简代码以通过测试
  3. 在测试通过的情况下进行重构
  4. 强制执行覆盖率(JaCoCo)

单元测试(JUnit 5 + Mockito)

@ExtendWith(MockitoExtension.class)
class MarketServiceTest {
  @Mock MarketRepository repo;
  @InjectMocks MarketService service;

  @Test
  void createsMarket() {
    CreateMarketRequest req = new CreateMarketRequest("name", "desc", Instant.now(), List.of("cat"));
    when(repo.save(any())).thenAnswer(inv -> inv.getArgument(0));

    Market result = service.create(req);

    assertThat(result.name()).isEqualTo("name");
    verify(repo).save(any());
  }
}

模式:

  • 准备-执行-断言(Arrange-Act-Assert)
  • 避免部分模拟;优先使用显式桩(stubbing)
  • 对于变体,使用 @ParameterizedTest

Web层测试(MockMvc)

@WebMvcTest(MarketController.class)
class MarketControllerTest {
  @Autowired MockMvc mockMvc;
  @MockBean MarketService marketService;

  @Test
  void returnsMarkets() throws Exception {
    when(marketService.list(any())).thenReturn(Page.empty());

    mockMvc.perform(get("/api/markets"))
        .andExpect(status().isOk())
        .andExpect(jsonPath("$.content").isArray());
  }
}

集成测试(SpringBootTest)

@SpringBootTest
@AutoConfigureMockMvc
@ActiveProfiles("test")
class MarketIntegrationTest {
  @Autowired MockMvc mockMvc;

  @Test
  void createsMarket() throws Exception {
    mockMvc.perform(post("/api/markets")
        .contentType(MediaType.APPLICATION_JSON)
        .content("""
          {"name":"Test","description":"Desc","endDate":"2030-01-01T00:00:00Z","categories":["general"]}
        """))
      .andExpect(status().isCreated());
  }
}

持久层测试(DataJpaTest)

@DataJpaTest
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
@Import(TestContainersConfig.class)
class MarketRepositoryTest {
  @Autowired MarketRepository repo;

  @Test
  void savesAndFinds() {
    MarketEntity entity = new MarketEntity();
    entity.setName("Test");
    repo.save(entity);

    Optional<MarketEntity> found = repo.findByName("Test");
    assertThat(found).isPresent();
  }
}

Testcontainers

  • 使用可复用的Postgres/Redis容器来模拟生产环境
  • 通过 @DynamicPropertySource 将JDBC URL注入Spring上下文

覆盖率(JaCoCo)

Maven 配置片段:

<plugin>
  <groupId>org.jacoco</groupId>
  <artifactId>jacoco-maven-plugin</artifactId>
  <version>0.8.14</version>
  <executions>
    <execution>
      <goals><goal>prepare-agent</goal></goals>
    </execution>
    <execution>
      <id>report</id>
      <phase>verify</phase>
      <goals><goal>report</goal></goals>
    </execution>
  </executions>
</plugin>

断言

  • 为提高可读性,优先使用 AssertJ(assertThat
  • 对于JSON响应,使用 jsonPath
  • 对于异常:assertThatThrownBy(...)

测试数据构建器

class MarketBuilder {
  private String name = "Test";
  MarketBuilder withName(String name) { this.name = name; return this; }
  Market build() { return new Market(null, name, MarketStatus.ACTIVE); }
}

CI 命令

  • Maven:mvn -T 4 testmvn verify
  • Gradle:./gradlew test jacocoTestReport

记住:保持测试快速、隔离且具有确定性。测试行为,而非实现细节。