作者:京东物流 杨建民
1.什么是Mock
Mock有仿照、假造的意义。Mock测验便是在测验过程中,关于某些不容易构造或许不容易获取的目标,用一个虚拟的目标来创立以便测验的测验办法。mock东西运用领域:
- 实在目标具有不确定的行为,发生不可猜测的效果。
- 实在目标很难被创立。
- 实在目标的某些行为很难被触发。
- 实在目标实际上还不存在。
MockIto和PowerMock是众多Mock结构中的两种,类似的还有:JMock,EasyMock,大多 Java Mock 库如 EasyMock 或 JMock 都是 expect-run-verify (希望-运转-验证)办法,而 Mockito 则运用更简略,更直观的办法:在执行后的互动中提问。运用 Mockito,你能够验证任何你想要的。而那些运用 expect-run-verify 办法的库,你常常被迫查看无关的交互。非 expect-run-verify 办法 也意味着,Mockito无需预备贵重的前期启动。他们的目标是透明的,让开发人员专注于测验选定的行为。
2.处理的问题
咱们在写单元测验时,总会遇到类似这些问题:
1.构造的入参,关于极值、反常鸿沟场景欠好复现,相关的逻辑测不到,只能依靠测验环境或预发跑,命运欠好或许要改好几次代码重启机器验证,费时费力;
2.依靠他人接口,或许需求他人帮忙测验环境数据库插数才能跑通;
3.依靠的他人的接口还没有开发完,为了不影响提测,如何完结单元测验?
4.编写的单元测验依靠测验数据库的数据,每次跑都要数据库改数?
5.对service层加了逻辑,跑单元测验本地验证的时分,由于种种原因,本地环境跑不起来,折腾半响跑起来验证完了,下次开发需求又遇到了另一个问题本地环境启动报错???
6.我就想dubug到某一行代码,可是逻辑杂乱,东拼西凑的参数便是走不到,自己看代码逻辑还要去问他人接口的返回值逻辑??(未完待续……)引进Mockito和PowerMock使得编写单元测验更轻松,更省时,更省力。
3.如何处理问题
3.1 运用mock的意义
简略说便是不管谁的本地环境,不管判别条件多么苛刻,不管本地数据库的测验数据被谁删了改了,不管他人接口的返回值逻辑多杂乱,不管自己代码逻辑多杂乱,都能独立的、可重复执行的、行等级覆盖的单元测验用例。
3.2 Mockito和PowerMock
一句话说Mockito和PowerMock。当所测逻辑里有静态东西类办法或私有办法咱们希望他返回特定值时(极值鸿沟、反常测验场景),咱们要用到PowerMock去弥补Mockito的缺乏,除此之外,用Mockito去写单测能完结咱们日常任务95%的场景。
3.3 运用Mcokito和PowerMock的最佳实践
3.3.1 引进pom文件
3.3.2 Mockito和PowerMock 两条通用语法
打桩:
when(XXxService.xxMethod(“希望入参”)).thenReturn(“希望出参”);验证:verify(XXxService).xxMethod(“希望入参”);
4.举例说明
4.1 SpringBoot项目下Mockito和PowerMock最佳实践
- classes: 指定要加载的类
- properties: 指定要设置特点
- @InjectMocks: 需求注入mock目标的Bean
- @MockBean或@Mock: 需求mock的Bean
import X;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.InjectMocks;
import org.mockito.MockitoAnnotations;
import org.powermock.api.mockito.PowerMockito;
import org.powermock.core.classloader.annotations.PowerMockIgnore;
import org.powermock.core.classloader.annotations.PrepareForTest;
import org.powermock.modules.junit4.PowerMockRunner;
/**
* 测验类A,调用服务B和一个静态东西类X
*/
@RunWith(PowerMockRunner.class)
@SpringBootTest(classes = {
A.class
})
@PowerMockIgnore({"javax.management.*"})
@PrepareForTest({X.class}) //mock 静态办法
public class ATest {
@InjectMocks
private A a;
@Mock
private B b;
@Before
public void setUp() throws Exception {
MockitoAnnotations.initMocks(this);
}
@Test
public void Test() {
when(b.someMethodB(any())).thenReturn(someThingB());
a.someMethodA(someThingA1(), someThingA2());
verify(b).someMethodB(any());
}
/**
* 反常鸿沟测验
*/
@Test
public void test_ExceptionTest() throws ParseException {
PowerMockito.mockStatic(X.class);
// 模仿反常抛出的场景
when(X.strToDate(anyString(), anyString())).thenThrow(ParseException.class);
when(X.convertLocalDateTime(any())).thenReturn(someThing());
when(b.someMethodB(any())).thenReturn(someThingB());
a.someThingA(someThingA1(), someThingA2());
verify(b).someMethodB(any());
}
高雅的mock能够考虑@spy,当然,mockito还有一些特性能够自行学习如:
5.遇到的一些问题及处理
- 打桩逻辑判别是通过equals办法判别的
- 测验的预期是抛出反常直接在注解上加:@Test(expected=BusException.class)
- 模仿的参数为null:Mockito.isNull()
- PowerMock mock静态和私有final会有一些格式区别
- PowerMockmock静态办法时也能够运用spy的办法使代码更高雅
- mock中发现,mock没有收效,能够测验晋级Mockito版别处理,另外与junit反射东西类结合运用,效果更佳。
- 涉及多层嵌套的运用场景,读者先思考”单元“选取是否合理,多层嵌套场景将@InjectMocks和@Spy(或@Mock)联合运用即可
结束语:
文章写于早些时分,目前有些较新技术涌入,如:Spock、TestableMock等,但上述技术依然适用于大型系统质量内建,读者可根据自身状况选择性选用。