当前位置:   article > 正文

SpringBoot教程(15) JUnit5 + Mockito @InjectMocks @Mock Stubbing_springboot @injectmocks

springboot @injectmocks

一、Mock测试介绍

1. 什么是Mock测试

Mock 测试就是在测试过程中,对于某些不容易构造(如 HttpServletRequest 必须在Servlet 容器中才能构造出来)或者不容易获取比较复杂的对象(如 JDBC 中的ResultSet 对象),用一个虚拟的对象(Mock 对象)来创建以便测试的测试方法。

2. Mock测试的常规步骤

  • Mock:创建出待测试的Mock对象
  • Stubbing:指定它的行为,return特定值或者抛出特定异常。when().thenReturn()或when().thenThrow()等
  • Verify:验证结果
    很多文章把Stubbing翻译成“打桩”或者“存根”,其实这样的翻译只会让人费解,所以不翻译,直接叫Stubbing,对程序员来说更好理解。

二、引入Mockito包

Mockito是比较出名且方便的Mock工具,它的API都比较直观,易于理解。
如果普通Java项目,不依赖SpringBoot,则直接引入

<dependency>
    <groupId>org.mockito</groupId>
    <artifactId>mockito-all</artifactId>
    <version>3.9.0</version>
    <scope>test</scope>
</dependency>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

如果是SpringBoot项目

<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-test</artifactId>
	<scope>test</scope>
</dependency>
  • 1
  • 2
  • 3
  • 4
  • 5

三、Mockito实践

0. 前提准备:待测试的业务代码

先创建几个非常简单的文件,这个结构大伙一看就懂。
在这里插入图片描述
DictTypeServiceImpl 的具体实现如下:
可以看到DictTypeServiceImpl中是依赖DictTypeDao的。

@Service
public class DictTypeServiceImpl extends CommonServiceImpl<DictType> implements DictTypeService {

	//这里依赖了DictTypeDao 
    @Autowired
    private DictTypeDao dictTypeDao;

    public DictType save(DictType dictType) {
        return dictTypeDao.save(dictType);
    }

    public void deleteById(String id){
        dictTypeDao.deleteById(id);
    }
    
    public Optional<DictType> getById(String id) {
        return dictTypeDao.findById(id);
    }

    public Iterable<DictType> listByQuery(QueryBuilder queryBuilder) {
        return search(queryBuilder, DictType.class);
    }

    public boolean existById(String id) {
        return dictTypeDao.existsById(id);
    }  
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27

1. 第一步:Mock

举例说明:
此时我就想写UT测试Service层的代码,不测试Dao层的代码,那就需要虚拟一个Dao层的对象(Mock对象)给Service用。
比如模拟Dao层中getById(“1”)的返回值,指定返回一个我们new的对象,而不是让Dao去查询数据库。
代码如下:

//我们自己定义一个返回值
DictType dictType = new DictType();
dictType.setId("1");
dictType.setCode("code1");
dictType.setName("name1");
Optional<DictType> dictTypeOptional = Optional.of(dictType);

//Stubbing,指定当执行到findById("1")时,返回上面定义的值,而不是执行真正的代码去查数据库
when(dictTypeDao.findById("1")).thenReturn(dictTypeOptional);
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

写这个UT时,可以创建3种对象:被测试对象、Mock对象、Spy对象

  • 被测试对象:就是你要测试的类,本博客中指DictTypeServiceImpl的对象。
  • Mock对象:虚拟出的对象,被测试对象依赖Mock对象。因为DictTypeServiceImpl中依赖DictTypeDao。本博客中Mock对象指DictTypeDao的对象。
  • Spy对象:Spy对象是一种特殊的Mock对象,它也可以作为被测对象的依赖对象。此时它和Mock对象的最大的区别:Mock对象的方法如果没有被Stubbing,调用时会返回相应对象的空值,而Spy对象的方法被调用时则会调用真实的代码逻辑。

1.1 @InjectMocks: 创建被测试对象

@InjectMocks的作用和@Autowired比较类似,但是它的成员变量将被@Mock和@Spy注解的字段注入。
DictTypeServiceImpl 的定义如下:

@Service
public class DictTypeServiceImpl extends CommonServiceImpl<DictType> implements DictTypeService {

	//这里依赖了DictTypeDao 
    @Autowired
    private DictTypeDao dictTypeDao;
	
	//省略其他方法
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

测试类中用@InjectMocks修饰在DictTypeServiceImpl上,那么@Mock或者@Spy修饰的对象会注入到@InjectMocks修饰的对象里。
注意@InjectMocks修饰在实现类上,而不是DictTypeService接口层,这个和@Autowired有不同。
在这里插入图片描述

1.2 @Mock:创建Mock对象

被测试的DictTypeServiceImpl中代码

public Optional<DictType> getById(String id) {
      return dictTypeDao.findById(id);
}
  • 1
  • 2
  • 3

用@Mock修饰在对象上,就可以实现Mock对象。

@SpringBootTest
public class MockitoWebTest {

    @InjectMocks
    DictTypeServiceImpl dictTypeService;

    @Mock
    private DictTypeDao dictTypeDao;

    @Test
    public void testMockObject() {

        //我们自己定义一个返回值
        DictType dictType = new DictType();
        dictType.setId("1");
        dictType.setCode("mock_code1");
        dictType.setName("mock_name1");
        Optional<DictType> dictTypeOptional = Optional.of(dictType);

        //Stubbing,指定当执行到findById("1")时,返回上面定义的值
        when(dictTypeDao.findById("1")).thenReturn(dictTypeOptional);

        Optional<DictType> dictTypeById_1 = dictTypeService.getById("1");
        Optional<DictType> dictTypeById_2 = dictTypeService.getById("2");

        System.out.println(dictTypeById_1.get().getCode());

        if (dictTypeById_2.isEmpty()){
            System.out.println("dictTypeById_2为空");
        } else {
            System.out.println(dictTypeById_2.get().getCode());
        }
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34

运行结果:

mock_code1
dictTypeById_2为空
  • 1
  • 2
  • Stubbing了dictTypeDao.findById(“1”),所以返回了我们自己预定义的值
  • 没有Stubbing dictTypeDao.findById(“2”),所以返回了空

2. 第二步:Stubbing

2.1 有返回的值的方法,模拟返回值

when(dictTypeDao.findById("1")).thenReturn(dictTypeOptional);
  • 1

2.2 有返回的值的方法,模拟抛出异常

when(dictTypeDao.findById("1")).thenThrow(new RuntimeException());
  • 1

2.3 无返回值void方法,模拟抛出异常

doThrow(new RuntimeException()).when(dictTypeDao).deleteById("1");
  • 1

2.4 doNothing

doNothing().when(dictTypeRepository).deleteById("1");
  • 1

2.5 匹配参数

上面的都是指定参数的,如果要匹配任意参数可以用anyString()、anyInt()、any(Class type)等等。

doThrow(new RuntimeException()).when(dictTypeRepository).deleteById(anyString());
  • 1

3. 第三步:Verify

verify()是验证方法执行的次数

when(mockedList.get(anyInt())).thenReturn("element");
System.out.println(mockedList.get(999));
System.out.println(mockedList.get(99));

verify(mockedList, times(1)).get(anyInt());
verify(mockedList).get(anyInt());//1次的也可以简写成这样

verify(mockedList, times(2)).get(anyInt());
verify(mockedList, atMost(2)).get(anyInt());
verify(mockedList, atLeast(2)).get(anyInt());
verify(mockedList, atLeastOnce()).get(anyInt());
verify(mockedList, never()).get(9);
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小丑西瓜9/article/detail/622079
推荐阅读
相关标签
  

闽ICP备14008679号