《Spring官方文档》第三部分 测试9-10章节

原文链接 译者:reed7

作为 Spring 的开发团队,我们鼓励在开发活动中引入测试驱动开发(TDD,Test-Driveng-Development)的行为,因此本文档接下来将涵盖 Spring 框架对集成测试的支持(以及 Spring 下单元测试的最佳实践)。

Spring 开发团队发现对控制反转(IoC)的正确运用可以使针对代码编写单元测试与集成测试变得更为容易(setter方法的存在,以及类里面恰当的构造器,使得测试代码在无需类似服务工厂等辅助工具的前提下,也能够方便地对各个类进行调用)。我们希望通过整整一章对 Spring 框架下测试相关主题的讲解,让读者也能对我们的以上发现表示赞同。

9. Spring 框架下的测试

测试乃企业级软件开发的重要组成部分之一。本章专注于讲解采用 IoC 原则进行编码而给单元测试带来的好处,以及 Spring 框架对集成测试的支持如何为测试带来帮助。(对企业开发中如何进行代码测试的详尽讨论不在本文档范围之内)

10. 单元测试

比起传统的 Java EE 开发方式,依赖注入可以弱化你的代码对容器的依赖。在基于 Junit 或 TestNG 的测试代码中,无需依赖于 Spring 或其他容器,你只需通过 new 操作符,便可以创建出组成你的应用程序的各种 POJO 对象。而通过 mock 对象(以及其它各种测试技术的综合运用),你可以将被测试的代码单独隔离开来进行测试。如果你在进行架构设计时遵循了 Spring 所推荐的模式,那么由此带来的诸如清晰的分层、组件化等等优点也会使你更容易地对你的代码进行单元测试。例如,通过 mock DAO 层或 Repositry 接口的实现,针对服务层对象的单元测试代码并不需要真正地访问持久层的数据。

真正的单元测试代码运行的非常快,因为并不需要一整套的运行时架构去支持测试的运转。所以在开发中强调真正的单元测试必须成为编码方法论中的一部分将会极大地提高你的生产力。

也许没有这一节内容的帮助你也能写出高效的针对 IoC 应用的单元测试,但是以下将要提到的 Spring 所提供的 mock 对象和测试支持类,在一些特定的场景中会对单元测试很有帮助。

10.1 Mock 对象

10.1.1 环境

org.springframework.mock.env 包含了对抽象 Environment 和 PropertySource 的 Mock 实现(参考 3.13.1 Bean的定义文件 和 3.13.3 PropertySource抽象)。MockEnvironment 和 MockPropertySource 对于编写针对依赖于环境相关属性的代码的,与容器无关的测试用例很有帮助。

10.1.2 JNDI

org.springframework.mock.jndi 包含了 JNDI SPI 的实现。这一实现可以让你为你的测试套件或独立应用配置起一个简单的 JNDI 环境。例如,如果测试代码和 Java EE 容器的 JDBC 数据源都绑定到同一个 JNDI 名字上,你就可以在测试中直接复用应用程序的代码和配置而不需做任何改动。

10.1.3 Servlet API

org.springframework.mock.web 包含了十分全面的 Servlet API mock 对象。这些对象在测试web 上下文,Controller 和 Filter 的时候很有用。由于这些 mock 对象是有针对性地为了与 Spring 的 Web MVC 框架共同使用而编写的,因此相比起诸如 EasyMock 这种动态mock 对象或 MockObjects 这种替代性的 Servlet API mock 对象,使用起来要更为方便。

假如是要把 Spring MVC 和 REST 控制器与 WebApplicationContext 配置结合起来进行一番彻底的集成测试,请使用 Spring MVC 测试框架 。

10.2 单元测试支持类

10.2.1 通用支持工具

org.springframework.test.util 包含了一些供单元测试和集成测试中使用的通用工具。

ReflectionTestUtils 是一组基于反射的方法集合。在测试包含如下一些测试用例的应用时,开发人员可以使用这些工具方法应对诸如更改一个常量的值,赋值一个非公有字段,调用一个非公有setter方法,或调用一个非公有的 配置 或 生命周期 回调方法等测试场景:

诸如 JPA 和 Hibernate 等广泛采用private或protected访问方式而非公有setter方法来访问domain entity属性的ORM(Object-Relational Mapping)框架
@Autowired,@Inject和@Resource等用于对private或protected字段,setter 方法和配置方法进行依赖注入的 Spring 注解。
@PostConstruct和@PreDestroy等用在生命周期回调方法上的注解。
AopTestUtils是一组 AOP 相关工具方法的集合。这些方法可以用于帮助获取隐藏于一重或多重 Spring 代理背后目标对象的引用。举个栗子,你使用 EasyMock 或 Mockito mock 了一个被包装在 Spring 代理之中的 bean,这时你可能需要对这个 mock 进行直接的访问,从而能够配置对此 mock 的期望行为并在稍后执行验证。关于 Spring 提供的核心 AOP 工具,请参考AopUtils和AopProxyUtils这两个类。

10.2.2 Spring MVC

org.springframework.test.web包含了ModelAndViewAssert类。你可以在用 Junit,TestNG 或任何测试框架编写的单元测试中使用这个类来帮助你跟 Spring MVC 框架的ModelAndView对象进行互动。

注:如果想要像测试 POJO 一样来测试你的 Spring MVC 控制器,你可以将ModelAndViewAssert与MockHttpServletRequestMockHttpSession等来自 Servlet API mocks 的 Mock 类结合使用。而假如是要把 Spring MVC 和 REST Controller 与WebApplicationContext配置结合起来进行一番彻底的集成测试,请使用 Spring MVC 测试框架。

原创文章,作者:ItWorker,如若转载,请注明出处:https://blog.ytso.com/tech/pnotes/100661.html

(0)
上一篇 2021年8月22日 00:49
下一篇 2021年8月22日 00:49

相关推荐

发表回复

登录后才能评论