Bootstrap

大白话单元测试之mock测试

一、什么是 Mock 测试

Mock通常是指,在测试一个对象A时,我们构造一些假的对象来模拟与A之间的交互,而这些Mock对象的行为是我们事先设定且符合预期。
这什么意思呢?
就是指在测试一个单元方法的时候,我们不想加载其他的类,默认其他类的功能是正常的,只测试这个方法。
好吧,我知道其实还是不太明白,那么请看示例

二、传统的单元测试Junit Test

首先看一下junit的单元测试是这样的,spring boot项目导入相关的依赖之后,
在测试类的上方注入两个注解,声明这是一个test类

@RunWith(SpringRunner.class)
@SpringBootTest(classes = {WebApplication.class})
public class BaseTest {
}

单元方法是下面这样的,通过@Resource注解将matchSourceScoreService注入进来

    @Resource
    private MatchSourceScoreService matchSourceScoreService;
    
    @Test
    public void saveMatchSourceScore(){
        MatchSourceScore matchSourceScore = matchSourceScoreService.selectById(6L);
        matchSourceScore.setStandardCode(10d);
        matchSourceScore.setUpdateBy("TEST_1221001");
        XyResultModel<Long> resultModel = matchSourceScoreService.saveMatchSourceScore(matchSourceScore);
        Assert.assertTrue(true);
    }

测试会发现,我们启动这个test类的时候会将整个项目启动起来,如果项目过大,每次单测一个test方法都会花费很长的时间

那么mock测试与junit测试有什么区别呢?

三、Mock测试(对比junit测试)

首先看一下这个测试类

@RunWith(MockitoJUnitRunner.class)
public class UpdateStandardCodeListenerTest {

    @InjectMocks
    private UpdateStandardCodeService updateStandardCodeService;
    @Mock
    private XyDrugStandardDao xyDrugStandardDao;

    private final StandardCodeExcelDTO standardCodeExcelDTO = new StandardCodeExcelDTO();

    @Before
    public void init() {
        standardCodeExcelDTO.setStandardCode("00000000000000");
        standardCodeExcelDTO.setApprovalNum("国食健字G20220206");
        standardCodeExcelDTO.setStrength("默认");
        standardCodeExcelDTO.setGenericName("百合康牌越橘叶黄素软胶囊");
        standardCodeExcelDTO.setXyEntManufacturerName("山东威海百合生物技术股份有限公司");
    }

    @Test
    public void testHandleStandardCodeByInputDTO1() throws CheckArgsException {
        when(xyDrugStandardDao.selectList(any())).thenReturn(null);
        updateStandardCodeService.handleStandardCodeByType(standardCodeExcelDTO, 4, "sys");
        verify(xyDrugStandardDao, times(1)).insert(any());
    }

看上去似乎和junit测试没啥区别,都需要在类上注入@RunWith注解,单元方法也是用@Test注解,
但是有变化的是:

  • 有一些方法用@InjectMocks注解,有些方法用@Mock注解。
  • 单元方法中有when。。。、verify。。。
  • 并不需要启动整个项目,只会单独运行这个方法

四、Mock详解

1.@InjectMocks和@Mock用法

这两个注解也是为了引用类中方法,将其托管给mock管理,
比如:

  • 我的目的是为了测试updateStandardCodeService类中的handleStandardCodeByType方法,那么就需要将其注解为@InjectMocks。
  • 那么被@Mock注解注入的方法是handleStandardCodeByType方法中用到了xyDrugStandardDao这个类中的selectList方法去查数据库,被这个注解注入后,并不会真的去发出sql语句去查询数据库,而是根据某种规则,返回特定的值。

handleStandardCodeByType方法:
可以看到方法中有selectList和insert两个数据库的交互操作

public void handleStandardCodeByType(StandardCodeExcelDTO standardCodeExcelDTO, int type, String operator) {
		Wrapper<XyDrugStandard> wrapper = new EntityWrapper<>();
        List<XyDrugStandard> xyDrugStandards = xyDrugStandardDao.selectList(wrapper);
        if (CollectionUtils.isEmpty(xyDrugStandards)) {
           xyDrugStandardDao.insert(xyDrugStandard);
        }
}

2.when用法

    when(xyDrugStandardDao.selectList(any())).thenReturn(null);

这就是一条规则,对于xyDrugStandardDao类的selectList方法,any()就是参数不管是什么,都会返回null。

3.verify用法

	verify(xyDrugStandardDao, times(1)).insert(any());

这是一条验证规则,验证xyDrugStandardDao的insert方法,不管参数是什么,有没有被执行1次,没有的话就报错,相当于断言Assert。

五、Mock其他常用用法

1.如何初始化数据

像我上面的那种方法,用@Before注解,初始化你的变量值,会在test方法执行前运行。

2.初始化service类中的Apollo配置

	@Test
    public void testHandleStandardCodeByInputDTO1() throws CheckArgsException {
    	setField("manufactureNameValidLength", 4);
    }

public void setField(String fieldName, Object value){
        Field field = XyDrugMasterCodeServiceImpl.class.getDeclaredField(fieldName);
        field.setAccessible(true);
        field.set(xyDrugMasterCodeService, value);
    }

比如我想给XyDrugMasterCodeServiceImpl类中的Apollo配置manufactureNameValidLength赋初值4。
就可以使用反射的方式,给配置赋初值

3.Assert.assertEquals用法

Assert.assertEquals("c", testList.get(0));

声明断言,testList的get(0)和字符串c相等。

还有一些其他用法可以参考官网https://site.mockito.org/

;