写点什么

Java-Mock 简化单元测试

用户头像
落日楼台H
关注
发布于: 2020 年 11 月 28 日
Java-Mock简化单元测试

单元测试目的

维基百科对单元测试的定义:
单元测试(英语:Unit Testing)又称为模块测试,是针对程序模块(软件设计的最小单位)来进行正确性检验的测试工作。程序单元是应用的最小可测试部件。在过程化编程中,一个单元就是单个程序、函数、过程等;对于面向对象编程,最小单元就是方法,包括基类(超类)、抽象类、或者派生类(子类)中的方法。
单元测试的目标是隔离程序部件并证明这些单个部件是正确的。
复制代码
  • 画外音:单元测试是比较细粒度的测试,是对接口、方法、函数的测试,目的是保障代码按照正确的方式去执行,提高代码质量。

单元测试实施原则

Mock 脱离数据库 + 不启动 Spring + 优化测试速度 + 不引入项目组件


单元测试不应该依赖数据,依赖外部服务或组件等,会对其他数据产生影响的情况。启动 Spring 容器,一般比较慢,可能会启动消息监听消费消息,定时任务的执行等,对数据产生影响。


Mock 测试就是在测试过程中,对那些当前测试不关心的,不容易构建的对象,用一个虚拟对象来代替测试的情形。


说白了:就是解耦(虚拟化)要测试的目标方法中调用的其它方法,例如:Service 的方法调用 Mapper 类的方法,这时候就要把 Mapper 类 Mock 掉(产生一个虚拟对象),这样我们可以自由的控制这个 Mapper 类中的方法,让它们返回想要的结果、抛出指定异常、验证方法的调用次数等等。


减少单元测试对外部的依赖和副作用,提高单元测试效率


  1. 不使用 @Autowired,@Resource, 需要启动 Spring 容器,测试速度慢,会产生副作用;

  2. 不使用 @SpringBootTest,@SpringBootTest(classes = Application.class), 这会启动整个 SpringBoot 服务

  3. 不应调用数据库,除非是做数据库操作相关的测试,虽然可配置事务回滚,但大多数情况下还是会产生脏数据等问题

  4. 使用 Assert 断言,用于判断某个特定条件下某个方法的行为,为了证明某段代码的执行结果和期望的一致

  • 画外音:单元测试应小而轻,提交测试效率,较少对外部的依赖,比如数据库、Spring 容器、网络服务等,而只关心我们自己的代码,通过 Mock 来解决对外部的依赖

Mockito 的使用

基本使用

  1. 使用静态方法 mock()

  2. 使用注解 @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 容器管理


  1. Spy 和 Mock 生成的对象不受 Spring 管理

  2. Spy 调用真实方法时,其它 bean 是无法注入的,要使用注入,要使用 SpyBean

  3. SpyBean 和 MockBean 生成的对象受 Spring 管理,相当于自动替换对应类型 bean 的注入,比如 @Autowired、@Resource 等注入

最佳实践

// 不使用 @SpringBootTest(classes = Application.class)@RunWith(SpringRunner.class)public class ExamAnswerComponentTest {
// 创建一个实例,会注入Mock变量 @InjectMocks private ExamAnswerComponent examAnswerComponent = new ExamAnswerComponentImpl();
// 相关操作会被Mock掉 @Mock private ExamAnswerCacheObjectiveDAO examAnswerCacheObjectiveDAO;
@Before public void setUp() { // 初始化Mock MockitoAnnotations.initMocks(this);
// given...willReturn 指定方法参数,模拟返回值 given(examAnswerCacheObjectiveDAO.selectByBizIdAndPaperAndQuestion(any(), any(), any())) .willReturn(new ExamAnswerCacheObjectivePO()); given(examAnswerCacheObjectiveDAO.insert(any())).willReturn(1); given(examAnswerCacheObjectiveDAO.updateUserAnswerById(any(), any())).willReturn(1); }

@Test public void saveOrUpdateAnswerCacheObjective() {
ExamAnswerCacheObjectivePO po = new ExamAnswerCacheObjectivePO(); po.setBizId(100000015L); po.setBizType(9); po.setUserAnswer("A"); po.setGroupPaperId(1000320L); po.setQuestionId(1000042L); po.setQuestionType(1);
int affect = examAnswerComponent.saveOrUpdateAnswerCacheObjective(po); System.out.println("affect = " + affect); Assert.assertTrue(affect > 0); }
}
复制代码

参考

https://www.codenong.com/cs106503150/


发布于: 2020 年 11 月 28 日阅读数: 876
用户头像

落日楼台H

关注

还未添加个人签名 2019.03.05 加入

还未添加个人简介

评论 (1 条评论)

发布
用户头像
简化测试如果可以用工具来解决,那么这么多年早NM解决了。在人、在流程、在文化,而不在工具
2020 年 11 月 29 日 17:27
回复
没有更多了
Java-Mock简化单元测试