mock模拟导致测试失败的解决方法

前言:在使用mock进行单元测试时,遇到了一个很奇怪的问题。比如一个测试类下有八个测试用例,其中有3个是使用了mock进行了模拟,同时彼此之间没有任何变量相联系。这八个测试用例单独一个个执行时没有任何报错,断言全部通过,但是当运行整个测试类时会发现有几个测试用例会报错,实际结果与预期结果有很大的偏差。

该问题主要是由于mock引起的,这里需要进行补充一下,一个测试用例的mock模拟,当你没有进行重置或者是还原替换,这个mock会一直伴随下去,被后续的测试用例使用,从而导致空指针等报错,或者是断言失败等问题出现。

在debug下能够看到一个含有mock的测试用例执行完后,下一个测试用例如果调用的是同一个方法类,会发现此时的方法类是被mock模拟的,也就是和上一个测试用例一样被mock所影响,从而导致这个测试用例出现错误。简单的来说就是:A中有mock,B中无mock,mock主要用于模拟C方法,A和B都会调用C。当测试类执行时,A先执行,模拟了C方法,使得A测试用例通过,之后执行B,但此时B所调用的C方法是A模拟的C方法,而不是原有的C方法,从而导致测试用例失败。

解决方法:

CareCourseCommentMapper careCourseCommentMapper = mock(CareCourseCommentMapper.class);
ReflectionTestUtils.setField(targetService,"targetMapper",careCourseCommentMapper);
        when(careCourseCommentMapper.selectById(anyLong())).thenReturn(careCourseComment);
        when(careCourseCommentMapper.deleteComment(anyLong())).thenThrow(new RuntimeException());


ReflectionTestUtils.setField(targetService,"targetMapper",targetMapper);



    @Autowired
    private CareCourseCommentMapper targetMapper;
    

若是前面模拟了mapper,那么在运行完后需要对mapper进行还原替换,确保mock只在这个测试用例中被模拟了,不会影响到其他的测试用例。即:如果前面进行了ReflectionTestUtils.setField进行了模拟替换和范围限制,并进行了模拟,那么在测试用例的结尾,我们需要对其进行还原,保证当测试用例运行完后,原有的方法类并没有受到影响,从而不会影响到后续的测试用例。

或者大家也可以试试reset方法,进行mock的重置替换,但该重置是重置到该测试用例的模拟处,这个方法主要用来对一个测试用例中要对mock模拟的方法进行二次修改,需要进行重置,大家也可以去试试。

你可能感兴趣的:(mock,单元测试,java,spring,boot,intellij,idea)