Mockito使用

前言

在Spring单元测试中,如果不想生成的对象受Spring管理的话,就可以使用Mock注解,为了减少单测与spring框架的耦合,并且尽量不使用真实对象进行模拟,可以使用@InjectMocks和@Mock注解

Mockito使用

生成@Mock、@Spy备注的mock对象,用

typescript 复制代码
@BeforeEach
	void setUp() {
		MockitoAnnotations.openMocks(this);
	}

定义一个实体类

vbnet 复制代码
@Data
public class UserQuery {

    
    private Long id;


    private String account;


    private String username;

    private Long deptId;


    private List<Long> roleIds;
    
    private String password;
    

}

定义一个service接口

scss 复制代码
public Boolean save(UserQuery userQuery) {
		UserDTO userDTO = new UserDTO();
		userDTO.setUsername(userQuery.getAccount());
		R<Boolean> result = remoteUserAdminService.isExist(userDTO);
		if (!Objects.equals(result.getCode(), CommonConstants.SUCCESS)) {
			throw new BussException("判断用户名是否存在错误");
		}
		if (result.getData()) {
			throw new BussException("用户名已存在");
		}
		userDTO.setNickName(userQuery.getUsername());
		result = remoteUserAdminService.user(userDTO);
		if (!Objects.equals(result.getCode(), CommonConstants.SUCCESS)) {
			throw new BussException("保存用户异常");
		}
		return result.getData();
	}

测试单元如下

scss 复制代码
@DisplayName("用户模块测试")
public class UserServiceImplTest {

	@InjectMocks
	private UserServiceImpl userService;

	@Mock
	private RemoteUserService remoteUserService;


	@Mock
	private RemoteUserAdminService remoteUserAdminService;

	@BeforeEach
	void setUp() {
		MockitoAnnotations.openMocks(this);
	}

	@Nested
	@DisplayName("新建用户单元测试")
	public class UserAddTest {

		@Test
		@DisplayName("保存用户,查询用户名是否存在,三方系统返回调用失败 => 返回异常")
		void save_1_test() {
			UserQuery userQuery = new UserQuery();
			userQuery.setUsername("aa");
			UserDTO userDTO = new UserDTO();
			userDTO.setUsername(userQuery.getAccount());
			R<Boolean> result = R.restResult(null, 1, "操作失败");
			Mockito.when(remoteUserAdminService.isExist(Mockito.any(UserDTO.class))).thenReturn(result);
			Assertions.assertThrows(BussException.class, () -> {
				userService.save(userQuery);
			});
			Mockito.verify(remoteUserAdminService, times(1)).isExist(Mockito.any(UserDTO.class));
		}

		@Test
		@DisplayName("保存用户,查询用户名是否存在,三方系统返回用户名已经存在 => 返回异常")
		void save_2_test() {
			UserQuery userQuery = new UserQuery();
			userQuery.setUsername("aa");			
			UserDTO userDTO = new UserDTO();
			userDTO.setUsername(userQuery.getAccount());
			R<Boolean> result = R.restResult(true, 0, "操作成功");
			Mockito.when(remoteUserService.isExist(Mockito.any(UserDTO.class))).thenReturn(result);
			Assertions.assertThrows(BussException.class, () -> {
				userService.save(userQuery);
			});
			Mockito.verify(remoteUserService, times(1)).isExist(Mockito.any(UserDTO.class));
		}

		@Test
		@DisplayName("保存用户,三方系统返回保存异常 => 返回异常")
		void save_3_test() {
			UserQuery userQuery = new UserQuery();
			userQuery.setUsername("aa");
			UserDTO userDTO = new UserDTO();
			userDTO.setUsername(userQuery.getAccount());
			R<Boolean> result = R.restResult(false, 0, "操作成功");
			Mockito.when(remoteUserService.isExist(Mockito.any(UserDTO.class))).thenReturn(result);
			R<Boolean> saveR = R.restResult(false, 1, "操作失败");
			Mockito.when(remoteUserService.user(Mockito.any(UserDTO.class))).thenReturn(saveR);
			Assertions.assertThrows(BussException.class, () -> {
				userService.save(userQuery);
			});
			Mockito.verify(remoteUserService, times(1)).isExist(Mockito.any(UserDTO.class));
			Mockito.verify(remoteUserService, times(1)).user(Mockito.any(UserDTO.class));
		}

		@Test
		@DisplayName("保存用户,三方系统返回成功 => 保存用户成功")
		void save_4_test() {
			UserQuery userQuery = new UserQuery();
			userQuery.setUsername("aa");
			UserDTO userDTO = new UserDTO();
			userDTO.setUsername(userQuery.getAccount());
			R<Boolean> result = R.restResult(false, 0, "操作成功");
			Mockito.when(remoteUserService.isExist(Mockito.any(UserDTO.class))).thenReturn(result);
			R<Boolean> saveR = R.ok(true, "操作成功");
			Mockito.when(remoteUserService.user(Mockito.any(UserDTO.class))).thenReturn(saveR);
			Boolean isSuccess = userService.save(userQuery);
			Mockito.verify(remoteUserService, times(1)).isExist(Mockito.any(UserDTO.class));
			Mockito.verify(remoteUserService, times(1)).user(Mockito.any(UserDTO.class));
			Assertions.assertTrue(isSuccess);
		}

	}

}

其中

Mockito.when

Mockito.when是模拟一个方法返回值,如

ini 复制代码
Mockito.when(remoteUserService.isExist(Mockito.any(UserDTO.class))).thenReturn(result);
			R<Boolean> saveR = R.ok(true, "操作成功");

模拟方法中调用isExist,返回result结果,Mockito.any(UserDTO.class)是传入一个UserDTO值

Mockito.verify

Mockito.verify(remoteUserService, times(1)),是验证方法中remoteUserService是不是被被用了一次

总结

在单元测试中,Mockito可以模拟出对象不受Spring框架影响,这样在没有Bean的注入和管理下,启动和验证就比较快,当然,写单元测试的时候,具体问题具体分析

相关推荐
我是若尘6 分钟前
Harness Engineering:2026 年 AI 编程的核心战场
前端·后端·程序员
lulu121654407820 分钟前
Claude Code Harness架构技术深度解析:生产级AI Agent工程化实践
java·人工智能·python·ai编程
阿里加多22 分钟前
第 1 章:Go 并发编程概述
java·开发语言·数据库·spring·golang
一 乐26 分钟前
物流信息管理|基于springboot + vue物流信息管理系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·论文·毕设·物流信息管理系统
2301_792674861 小时前
java学习day29(juc)
java·开发语言·学习
IT_陈寒1 小时前
折腾一天才明白:Vite的热更新为什么偶尔会罢工
前端·人工智能·后端
希望永不加班2 小时前
SpringBoot 自动配置类加载顺序与优先级
java·spring boot·后端·spring·mybatis
纸鸢|2 小时前
从“一锤子买卖“到“持续价值共生“:物联网软件产品如何做到“叫好又叫座“
java·物联网·struts
AI茶水间管理员2 小时前
学习ClaudeCode源码之Agent核心循环
前端·人工智能·后端
Reart3 小时前
从0解构tinyWeb项目--(Day:2)
javascript·后端·架构