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文件配置的方式。

04-15 14:24