这个章节包含了 Spring 对集成测试的支持和单元测试的最佳实践。Spring 团队提倡测试驱动开发(TDD),他们已发现通过正确使用控制反转(IoC)的确可以让单元测试和集成测试变得更简易(已有 setter 和合适的构造器,因此更容易结合到测试中,而不需要设置服务注册器等相似结构)。
1. Spring 测试简介
测试是企业级软件开发必备的一部分。这个章节关注于 Ioc 原则给单元测试带来的价值,以及 Spring 框架对集成测试支持的优点。(企业中一个完整测试处理的范围会多于此参考手册)。
2. 单元测试
相对于传统 Java EE 开发,依赖注入会让你的代码更少地依赖于容器。即使不使用 Spring 或其他容器,通过 JUnit 或 TestNG 以及new
实例化的对象,构成应用的 POJO 类也应是可测试的。你可以使用 “模拟对象”,与其他测试技术相结合,来隔离地测试你的代码。如果你遵循 Spring 推荐结构,那么简洁的分层和组件化能让你的代码库更易于单元测试。比如,你能在去除或模拟 DAO 接口的情况下测试 service 层对象,无需访问持久层数据。
2.1. 模拟对象(Mock Objects)
Spring 包括若干用于模拟对象的包:
- Environment
- JNDI
- Servlet API
- Spring Web Reactive
2.1.1. Environment
org.springframework.mock.env
包含对环境和配置源的模拟实现(可参考 Spring Bean 定义和配置源抽象类)。环境和配置源模拟很适合开发 无容器测试 和 依赖特定环境 的代码。
2.1.2. JNDI
org.springframework.mock.jndi
包含了 JNDI(Java命名和目录接口)的实现,你可以为测试套件或独立应用设置简易的 JNDI 环境。例如,如果你的 JDBC 数据源使用了和测试代码中一样的 JNDI 命名风格,你就能在测试场景中复用应用代码而无需修改。
2.1.3. Servlet API
org.springframework.mock.web
包含了全面的 Servlet API 对象模拟,可用于测试 Web 上下文,Controller 接口,Filter 过滤器。这些模拟对象关注于 Spring MVC 框架,并且相比动态模拟对象(如 EasyMock)更加易用。
从 Spring 5.0 起, org.springframework.mock.web 中的对象模拟都是基于 Servlet 4.0 API 的。
2.1.4. Spring Web Reactive
org.springframework.mock.http.server.reactive
中包含了模拟Servlet 的 HTTP 请求和响应的实现,可用于 WebFlux 应用。org.springframework.mock.web.server 包括了模拟 ServerWebExchange 的请求和响应的支持。
继承同一基类的 MockServerHttpRequest 和 MockServerHttpResponse 有相同的服务端实现和行为。例如,一个模拟请求在创建后就是不可变的,但可以通过调用 mutate() 方法,来创建可修改实例。
为了让模拟响应正确地输出并返回完成句柄,默认通过 cache().then() 使用了流,能够缓存数据并用于测试断言。应用可以给响应设置自定义输出函数(比如无限的输出流)。
模拟响应和请求的 WebTestClient 提供了无需 HTTP 服务端情况下,测试 WebFlux 应用的支持。它也可以用于运行中服务器的端到端测试。
2.2. 单元测试支持类
Spring 提供了若干可用于单元测试的类,分为两类:
- 通用测试工具类
- Spring MVC 测试工具类
2.2.1. 通用测试工具类
org.springframework.test.util
包含了一些一般用途的单元测试和集成测试工具类。
ReflectionTestUtils 是基于反射的工具方法集。可以用在需要修改常量、修改非公共属性、调用非公共方法、调用生命周期方法的测试场景。如:
- ORM 框架(如 JPA 和 Hibernate)下,领域实体中私有或受保护的字段。
- Spring 的注解支持(如 @Autowired,@Inject,@Resource),可以提供对私有或受保护字段、配置方法的依赖注入。
- 使用如 @PostConstruct 和 @PreDestroy 作为生命周期的调用。
AopTestUtils是AOP切面有关的工具方法集。你可以用这些方法来获得 Spring 代理下目标对象的引用。例如,如果你使用 EasyMock 或 Mockito 配置了一个一个动态模拟对象,它是被 Spring 代理的,而你可能需要直接访问内在对象设置预期、执行验证。对于 Spring 的核心 AOP 工具类,可参考 AopUtils
和 AopProxyUtils
。
2.2.2. Spring MVC 测试工具类
org.springframework.test.web
包含了ModelAndViewAssert
,可结合 Junit,TestNG 或其他框架来为模型和视图做单元测试。