mockito+junit完成单元测试

一:单元测试的特点

  • 配合断言使用(可以杜绝System.out)
  • 可以重复执行
  • 不依赖环境
  • 不会对数据产生影响
  • spring的上下文环境不是必须的
  • 一般都需要配合mock类框架来实现

二:常用的mock类框架

mockito

官网:Mockito framework site

另外现在像powermock和JMockito现在都不常用了;

三:Mockito的单独使用

(1)mock对象和spy对象

|------|--------|--------------|------|----------|
| | 方法插桩 | 方法不插桩 | 作用对象 | 最佳实践 |
| mock | 执行插桩逻辑 | 返回mock对象的默认值 | 类、接口 | 被测试类或其依赖 |
| spy | 执行插桩逻辑 | 调用真实方法 | 类、接口 | 被测试类 |

(2)初始化mock/spy对象的方式

|--------|----------------------------------------------|----------------------------|---------------------------------------------|
| | 方法一 | 方法二 | 方法三 |
| junit4 | @RunWith(MockitoJUnitRunner.class)+@Mock等注解 | Mockito.mock(X.class)等静态方法 | MockitoAnnotations.openMocks(this)+@Mock等注解 |
| junit5 | @ExtendWith(MockitoExtension.class)+@Mock等注解 | Mockito.mock(X.class)等静态方法 | MockitoAnnotations.openMocks(this)+@Mock等注解 |

四:具体实例操作

Controller层:

java 复制代码
@Slf4j
@RestController
@Validated
public class UserController{
    @Resource
    private UserService userService;

    @GetMapping("/selectById")
    public UserVO selectById(@NotNull Long userId){
        return userService.selectById(userId);
    }

    @PostMapping("/add")
    public String add(@RequestBody @Validated UserAddReq addReq){
        userService.add(addReq.getUsername(),addReq.getPhone(),addReq.getfeatures());
        return ok;
    }
}

方法一:@ExtendWith(MockitoExtension.class)+@Mock等注解

java 复制代码
@ExtendWith(MockitoExtension.class)
public class InitMockSpyMethod1{
    @mock
    private UserService mockUserService;
    @Spy
    private UserService spyUserService;

    @Test
    public void test1(){
        //true     判断某对象是不是mock对象
        System.out.println("Mockito.mockingDetails(mockUserService).isMock() = " + Mockito.mockingDetails(mockUserService).isMock());

        //false     判断某对象是不是apy对象
        System.out.println("Mockito.mockingDetails(mockUserService).isMock() = " + Mockito.mockingDetails(mockUserService).isMock());

        //true     判断某对象是不是spy对象
        System.out.println("Mockito.mockingDetails(spyUserService).isSpy() = " + Mockito.mockingDetails(spyUserService).isSpy());

        //true     判断某对象是不是spy对象,因为spy是一种特殊的mock(spy对象是另一种不同类型的mock对象)
        System.out.println("Mockito.mockingDetails(spyUserService).isMock() = " + Mockito.mockingDetail(spyUserService).isMock());
    }
}

spy对象是一种特殊的mock对象

方法二:Mockito.mock(X.class)等静态方法

java 复制代码
public class InitMockSpyMethod1{
    private UserService mockUserService;
    private UserService spyUserService;

    @BeforeEach
    public void init(){
        mockUserService = Mockito.mock(UserService.class);
        spyUserService =  Mockito.spy(UserService.class);
    }
    
    @Test
    public void test1(){
        System.out.println("Mockito.mockingDetails(mockUserService).isMock() = " + Mockito.mockingDetails(mockUserService).isMock());
        System.out.println("Mockito.mockingDetails(mockUserService).isSpy() = " + Mockito.mockingDetails(mockUserService).isSpy());
        System.out.println("Mockito.mockingDetails(spyUserService).isMock() = " + Mockito.mockingDetails(spyUserService).isMock());
    }
}

方法三:MockitoAnnotations.openMocks(this)+@Mock等注解

java 复制代码
public class InitMockSpyMethod1{
    @mock
    private UserService mockUserService;
    @Spy
    private UserService spyUserService;

    @BeforeEach
    public void init(){
        MockitoAnnotations.openMocks(this);
    }
    
    @Test
    public void test1(){
        //true     判断某对象是不是mock对象
        System.out.println("Mockito.mockingDetails(mockUserService).isMock() = " + Mockito.mockingDetails(mockUserService).isMock());

        //false     判断某对象是不是apy对象
        System.out.println("Mockito.mockingDetails(mockUserService).isMock() = " + Mockito.mockingDetails(mockUserService).isMock());

        //true     判断某对象是不是spy对象
        System.out.println("Mockito.mockingDetails(spyUserService).isSpy() = " + Mockito.mockingDetails(spyUserService).isSpy());

        //true     判断某对象是不是spy对象,因为spy是一种特殊的mock(spy对象是另一种不同类型的mock对象)
        System.out.println("Mockito.mockingDetails(spyUserService).isMock() = " + Mockito.mockingDetail(spyUserService).isMock());
    }
}

五:参数匹配

参数匹配指的是:通过方法签名(参数)来指定哪些方法调用需要处理被处理(插桩、verify验证)

对于mock对象不会调用真实方法,直接返回mock对象的默认值;

默认值(int)、null(UserVO)、空集合(list)

六:方法插桩

指定调用某个方法时的行为(stubbing),达到相互隔离的目的

java 复制代码
/**
*测试插桩时的参数匹配
*/
@Test
public void test2(){
    UserUpdateReq userUpdateReq1 = new UserUpdateReq();
    userUpdateReq1.setId(1L);
    userUpdateReq1.setPhone("1L");
    //指定参数为userUpdateReq1时调用mockUserService.modifyById(userUpdateReq1);
    Mockito.doReturn(99).when(mockUserService).modifyById(userUpdateReq1);
    int result1 = mockUserService.modifyById(userUpdateReq1);
    //运行结果为99
    System.out.println("result1 = " + result1);
    
    UserUpdateReq userUpdateReq2 = new UserUpdateReq();
    userUpdateReq2.setId(2L);
    userUpdateReq2.setPhone("2L");
    int result2 = mockUserService.modifyById(userUpdateReq2);
    //运行结果为0
    System.out.println("result2 = " + result2);
}

总结

是告诉mockito当传入的是下面的参数这个类型时,才对其进插桩,若不是这个值,则不用对它进行插桩

若想要拦截某种类型的任意对象,则需要运用到:

ArgumentMatchers.any拦截UserUpdateReq类型的任意对象

校验

add方法调用一次,校验通过。

java 复制代码
  private UserService mockUserService;

    @Test
    public void test4(){
        List<String> features = new ArrayList<>();
        mockUserService.add("实验","123",festures);

        //校验参数为"乐之者","123",festures,features的add方法调用了1次
        Mockito.verify(mockUserService,Mockito.times(2)).add("实验","123",festures);

        //报错  ,要么都用要么就都别用
       // Mockito.verify(mockUserService,Mockito.times(2)).add("实验","123",festures);

        //此时可以校验通过
         Mockito.verify(mockUserService,Mockito.times(2)).add(anyString(),anyString(),anyList());
    }
    
}

但是有一点需要牢记的,除了any,还有(anyLong,anyString...),注意他们都不包括null,如果传null,还是不能被匹配的。

通过插桩,指定方法的返回值:

void返回值方法插桩 :

插桩的两种方式:

多次插桩:

其中when(mockList.size()).thenReturn(1).thenReturn(2).thenReturn(3)可以简写为when(mockList.size()).thenReturn(1,2,3);

thenAnswer指定插桩逻辑:

执行真正的原始方法:

verify的使用:

@InjectMocks注解的使用

1.被@InjectMocks标注的属性必须是实现类,因为mockito会创建对应的实例对象,默认创建的对象就是未经过mockito处理的普通对象。因此常配合@spy注解使其变成默认调用真实方法的mock对象。

2.mockito会使用spy,最终的结果就是,会把userFeatureService给注入到InjectMocks标注的变量所对应的对象里面去。

注入的原理

相关推荐
卓码软件测评4 小时前
CMA/CNAS双资质软件测评机构【Apifox高效编写自动化测试用例的技巧和规范】
测试工具·ci/cd·性能优化·单元测试·测试用例
回眸&啤酒鸭7 小时前
【回眸】Tessy 单元测试软件使用指南(五)进阶报错之解决指南(含泪整理)
单元测试
小二·7 小时前
前端测试体系完全指南:从 Vitest 单元测试到 Cypress E2E(Vue 3 + TypeScript)
前端·typescript·单元测试
无心道人c9 小时前
SonarQube7.6实现C#自定义规则
单元测试·自动化·sonar
汽车仪器仪表相关领域9 小时前
多气精准检测,全场景适配——NHA-506/406型汽车排放气体测试仪项目实战分享
大数据·功能测试·单元测试·汽车·可用性测试·安全性测试
姓蔡小朋友1 天前
LUA脚本
开发语言·junit·lua
卓码软件测评1 天前
第三方软件国产化测评机构【API验收测试,除了Postman,还有什么推荐工具?】
测试工具·ci/cd·性能优化·单元测试·测试用例·postman
shughui1 天前
JMeter(二):什么是jmeter参数化?为什么需要它?如何使用
测试工具·jmeter·性能优化·单元测试
卓码软件测评2 天前
首版次软件认证测试机构【Apifox GraphQL支持详解:查询、变更和订阅】
测试工具·ci/cd·性能优化·单元测试·测试用例
llilian_162 天前
如何挑对一款数字式频率计数器?时间间隔测量仪 国产频率计
单片机·测试工具·单元测试·自动化