当前位置: 首页 > news >正文

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等注解

 四:具体实例操作

Controller层:

@Slf4j
@RestController
@Validated
public class UserController{@Resourceprivate 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等注解

@ExtendWith(MockitoExtension.class)
public class InitMockSpyMethod1{@mockprivate UserService mockUserService;@Spyprivate UserService spyUserService;@Testpublic 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)等静态方法

public class InitMockSpyMethod1{private UserService mockUserService;private UserService spyUserService;@BeforeEachpublic void init(){mockUserService = Mockito.mock(UserService.class);spyUserService =  Mockito.spy(UserService.class);}@Testpublic 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等注解

public class InitMockSpyMethod1{@mockprivate UserService mockUserService;@Spyprivate UserService spyUserService;@BeforeEachpublic void init(){MockitoAnnotations.openMocks(this);}@Testpublic 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),达到相互隔离的目的

/**
*测试插桩时的参数匹配
*/
@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);//运行结果为99System.out.println("result1 = " + result1);UserUpdateReq userUpdateReq2 = new UserUpdateReq();userUpdateReq2.setId(2L);userUpdateReq2.setPhone("2L");int result2 = mockUserService.modifyById(userUpdateReq2);//运行结果为0System.out.println("result2 = " + result2);
}

总结

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

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

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

 校验

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

  private UserService mockUserService;@Testpublic 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标注的变量所对应的对象里面去。

注入的原理

http://www.lryc.cn/news/473116.html

相关文章:

  • 基于web的便捷饭店点餐小程序的设计与实现(lw+演示+源码+运行)
  • CUDA环境安装终极指南——Linux(其它系统也一样)
  • 安卓开发之登录页面(跳转版)
  • solidworks学习6吊环-20241030
  • 数据结构和算法-动态规划(3)-经典问题
  • Java算法-一维前缀和与差分
  • Elasticsearch 安装教程:驾驭数据海洋的星际导航仪
  • 【解决方案】微信小程序如何使用 ProtoBuf 进行 WebSocket 通信
  • 独立游戏开发者面临的挑战与困境
  • KVM 虚拟机Anolis OS 8.9 下利用宝塔面板中的 Docker 配置 Nextcloud + onlyoffice
  • 串口扫盲TTL,TX/TR/GND
  • Python酷库之旅-第三方库Pandas(181)
  • Python数据分析NumPy和pandas(十七、pandas 二进制格式文件处理)
  • matlab计算相关物理参数
  • nmcli、ip、ifcfg配置网络区分方法
  • 第四届智能电力与系统国际学术会议(ICIPS 2024)
  • 区块链样题第4套解析 后端应用开发部分
  • C语言实现408考研真题2016年43题
  • 2024年,Rust开发语言,现在怎么样了?
  • 三种网络配置方法nmcli、ip、ifcfg文件
  • AES_ECB算法C++与Java相互加解密Demo
  • H7-TOOL自制Flash读写保护算法系列,为兆易创新GD32E23X制作使能和解除算法,支持在线烧录和脱机烧录使用(2024-10-29)
  • FFmpeg 深度教程音视频处理的终极工具
  • Java程序设计:spring boot(13)——全局异常与事务控制
  • 金和OA-C6 ApproveRemindSetExec.aspx XXE漏洞复现(CNVD-2024-40568)
  • Redis集群及Redis存储原理
  • 基于Springboot的图书个性化推荐系统【源码】+【论文】
  • 科普 | 子母钟系统是什么?网络时钟同步的重要性?
  • 批量删除redis数据【亲测可用】
  • Vuestic 数据表格 使用demo