1 Mockito简介
1.1 Mockito是什么
Mockito是一个简单的流行的Mock框架。它允许你创建和配置mock对象。使用Mockito可以明显的简化对外部依赖的测试类的开发。一般使用 Mockito 需要执行下面三步:
模拟并替换测试代码中外部依赖;
执行测试代码;
验证测试代码是否被正确的执行。
1.2 Mock是什么
Mock测试就是在测试过程中,对某些不容易构造或者不容易获取的对象,用一个虚拟的Mock对象来创建以便测试的测试方法。
Mock最大的功能是帮你把单元测试的耦合分解开,如果你的代码对另一个类或者接口有依赖,它能够帮你模拟这些依赖,并帮你验证所调用的依赖的行为。
2 Mockito在Spring框架中的使用
2.1 依赖包
<dependency>
<groupId>org.mockito</groupId>
<artifactId>mockito-core</artifactId>
<version>
2.21
.
0
</version>
<scope>test</scope>
</dependency>
|
Mockito在Spring可以通过配置文件和注解的方式使用
2.2 通过xml配置文件使用Mockito
Mockito创建方式有两种:mock,spy。mock为一个interface提供一个虚拟的实现,spy为object加一个动态代理,实现部分方法的虚拟化,但是需要赋予一个instance。
Mockito提供了factory的方法用来创建mock和spy。
假设构造系统中有serviceA,serviceB,serviceC,其中serviceA依赖serviceB依赖serviceC。serviceC是最基本的。现在需要对其中serviceA进行测试,但其serviceC需要依赖于外部环境,而这个环境需要复杂而且不稳定的数据库。这时需要mock或spy掉serviceC。
注:无论是spring框架还是spring boot框架,如果集成了MyBatis,因Mybatis中Mapper基于动态代理实现,则其对应的mapper接口只能基于xml文件配置方式注入mock或spy。
具体解释参考:https://sq.163yun.com/blog/article/169561874192850944
第一步:在配置文件中注入mock:
<bean id=
"serviceC"
class
=
"org.mockito.Mockito"
factory-method=
"mock"
>
<constructor-arg value=
"com.x.y.x.ServiceC"
></constructor-arg>
</bean>
|
或注入spy
<bean id=
"serviceCInst"
class
=
"serviceCInstance"
>
</bean>
<bean id=
"serviceC"
class
=
"org.mockito.Mockito"
factory-method=
"spy"
>
<constructor-arg ref=
"serviceCInst"
></constructor-arg>
</bean>
|
spy需要获得一个实例。
第二步:
在测试用例中@Resource或@Autowired引入serviceC。
2.3 通过注解使用Mockito
以代码举例
@Component
public
class
OrderCreate {
@Resource
private
OrderHelper orderHelper;
public
void
create() {
System.out.println(getAmt());
System.out.println(orderHelper.resolve());
}
public
int
getAmt() {
return
10
;
}
}
|
@Component
public
class
OrderHelper {
public
String
resolve() {
return
"resolve order"
;
}
}
|
测试类:
public
class
MockSpringTest {
@InjectMocks
private
OrderCreate orderCreate;
@Mock
private
OrderHelper orderHelper;
// 此mock将被注入到orderCreate
@Before
public
void
initMocks() throws Exception {
MockitoAnnotations.initMocks(
this
);
}
@Test
public
void
create() {
System.out.println(
"start mock..."
);
when(orderHelper.resolve()).thenReturn(
"null"
);
orderCreate.create();
}
}
|
运行结果:
start mock...
10
null
|
mock() / @Mock:创建一个mock;
spy() / @Spy:创建一个spy,提供了一种对真实对象操作的方法;
@InjectMocks:创建一个实例。被@Mock(或@Spy)注解创建的mock将被注入到用该实例中。注意: @InjectMocks标注的属性不能使用接口。
3 Mockito在Spring boot 框架中的使用
3.1 Spring Boot自带测试模块
Spring boot自身提供很多有用的工具类和注解用于测试应用,主要分两个模块:spring-boot-test包含核心组件,spring-boot-test-autoconfigure为测试提供自动配置。开发者只需要引用spring-boot-starter-test即可。它提供的测试模块中包含了Mockito。
Spring boot使用@MockBean和@SpyBean来定义Mockito的mock和spy。SpringBoot提供的@MockBean注解,用于为Applicatio nContext中的bean定义一个mock,你可以使用该注解添加新beans,或替换已存在的bean定义。该注解可直接用于测试类,也可用于测试类的字段,或用于@Configuration注解的类和字段。当用于字段时,创建mock的实例也会被注入。Mock beans每次调用完测试方法后会自动重置。
以代码举例:
@Component
public
class
MethodTest {
public
String
One(boolean flag){
System.err.println(
"coming........."
);
String
d = Two(flag);
Three();
System.err.println(
"result data is "
+d);
return
d;
}
public
String
Two(boolean flag){
System.err.println(
"coming.........two"
);
if
(flag){
throw
new
IllegalAccessError();
}
return
"two"
;
}
public
void
Three(){
System.err.println(
"coming.........three"
);
}
}
|
测试类 @SpyBean
@RunWith(SpringRunner.
class
)
@SpringBootTest(classes = DemoApplication.
class
)
public
class
SpyTest{
@SpyBean
private
MethodTest spyTest;
@Test
public
void
test3(){
when(spyTest.Two(
false
)).thenReturn(
"test"
);
System.err.println(spyTest.One(
false
));
}
}
|
运行结果:
coming.........two
coming.........
coming.........three
result data
is
test
test
|
测试类 @MockBean
@RunWith(SpringRunner.
class
)
@SpringBootTest(classes = DemoApplication.
class
)
public
class
MockTest{
@MockBean
private
MethodTest mockTest;
@Test
public
void
test3(){
when(mockTest.One(
false
)).thenReturn(
"test"
);
System.err.println(mockTest.One(
false
));
}
}
|
运行结果:
test
|
3.2 Spring Boot使用原生Mockito
使用方式见第2部分。
4 总结
为了统一springBoot和spring项目对Mockito的配置方式,同时解决框架中Mybaties之Mapper动态代理的mock场景,建议统一使用xml文件配置的方式。