Java-Mock 简化单元测试
单元测试目的
画外音:单元测试是比较细粒度的测试,是对接口、方法、函数的测试,目的是保障代码按照正确的方式去执行,提高代码质量。
单元测试实施原则
Mock 脱离数据库 + 不启动 Spring + 优化测试速度 + 不引入项目组件
单元测试不应该依赖数据,依赖外部服务或组件等,会对其他数据产生影响的情况。启动 Spring 容器,一般比较慢,可能会启动消息监听消费消息,定时任务的执行等,对数据产生影响。
Mock 测试就是在测试过程中,对那些当前测试不关心的,不容易构建的对象,用一个虚拟对象来代替测试的情形。
说白了:就是解耦(虚拟化)要测试的目标方法中调用的其它方法,例如:Service 的方法调用 Mapper 类的方法,这时候就要把 Mapper 类 Mock 掉(产生一个虚拟对象),这样我们可以自由的控制这个 Mapper 类中的方法,让它们返回想要的结果、抛出指定异常、验证方法的调用次数等等。
减少单元测试对外部的依赖和副作用,提高单元测试效率
不使用 @Autowired,@Resource, 需要启动 Spring 容器,测试速度慢,会产生副作用;
不使用 @SpringBootTest,@SpringBootTest(classes = Application.class), 这会启动整个 SpringBoot 服务
不应调用数据库,除非是做数据库操作相关的测试,虽然可配置事务回滚,但大多数情况下还是会产生脏数据等问题
使用 Assert 断言,用于判断某个特定条件下某个方法的行为,为了证明某段代码的执行结果和期望的一致
画外音:单元测试应小而轻,提交测试效率,较少对外部的依赖,比如数据库、Spring 容器、网络服务等,而只关心我们自己的代码,通过 Mock 来解决对外部的依赖
Mockito 的使用
基本使用
使用静态方法 mock()
使用注解 @Mock 标注
如果使用 @Mock 注解, 必须去触发所标注对象的创建. 可以使用 MockitoRule 来实现. 它调用了静态方法 MockitoAnnotations.initMocks(this) 去初始化这个被注解标注的字段.或者也可以使用 @RunWith(MockitoJUnitRunner.class).
“when thenReturn”和”when thenThrow”
模拟对象可以根据传入方法中的参数来返回不同的值, when(….).thenReturn(….)方法是用来根据特定的参数来返回特定的值.
我们也可以使用像 anyString 或者 anyInt anyLong any 这样的方法来定义某个依赖数据类型的方法返回特定的值.
“doReturn when” 和 “doThrow when”
doReturn(…).when(…)的方法调用和 when(….).thenReturn(….)类似.对于调用过程中抛出的异常非常有用.而 doThrow 则也是它的一个变体.
常用注解
@Mock:对函数的调用均执行 mock(即虚假函数),不执行真正部分。
@Spy:对函数的调用均执行真正部分。
@InjectMocks:创建一个实例,简单的说是这个 Mock 可以调用真实代码的方法,使用 @Mock(或 @Spy)注解创建的 mock 将被注入到用该实例中。
Mockito 中的 Mock 和 Spy 都可用于拦截那些尚未实现或不期望被真实调用的对象和方法,并为其设置自定义行为。二者的区别在于 Mock 不真实调用,Spy 会真实调用。
@MockBean: 功能同 @Mock, 只是会将实例放入 Spring 容器管理
@SpyBean: 功能同 @Spy, 只是会将实例放入 Spring 容器管理
Spy 和 Mock 生成的对象不受 Spring 管理
Spy 调用真实方法时,其它 bean 是无法注入的,要使用注入,要使用 SpyBean
SpyBean 和 MockBean 生成的对象受 Spring 管理,相当于自动替换对应类型 bean 的注入,比如 @Autowired、@Resource 等注入
最佳实践
参考
https://www.codenong.com/cs106503150/
版权声明: 本文为 InfoQ 作者【落日楼台H】的原创文章。
原文链接:【http://xie.infoq.cn/article/eb5352ade963a1671d646b2ac】。文章转载请联系作者。
评论 (1 条评论)