单元测试进阶Mockito工具
一、概述
1.Mockito是什么工具
Mockito框架可以创建和配置mock对象,简化具有外部对象依赖的类的测试开发。
2.项目中引入Mockito
Maven引入依赖
<!-- https://mvnrepository.com/artifact/org.mockito/mockito-core -->
<dependency>
<groupId>org.mockito</groupId>
<artifactId>mockito-core</artifactId>
</dependency>
// 初始化Mockito,或者junit4中使用@Runwith(MockitoJunitRunner.class) MockitoAnnotations.openMocks(this);
二、思想及使用方法
1.@TestInstance注解
在使用Mockito之前需要初始化,在junit4中可以使用@Runwith(MockitoJunitRunner.class)设置运行器。但是junit5中暂未找到设置运行器的注解,因此我在@BeforeAll注解中将其初始化。
随之而来的问题是,初始化需要使用到this,并且@BeforeAll注解的方法需要是static方法。这两个是冲突的,因此需要在类上加上@TestInstance注解,如下
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
2.@InjectMocks注解
该注解用于被测试的类上,被它注解的类,方法会真实的执行。
当该测试类遇到使用@Mock注解的依赖时,会使用被虚拟后的对象
3.@Mock注解
使用@Mock注解的类会创建一个完全虚拟的实例,方法和属性都是空值,不会真实执行。
用于创建被测试类的依赖使用,如对Dao或Service的Mock
三、示例
1.对Service的测试,对Dao的Mock
当我们做好Mockito的初始化,及定义好需要@InjectMocks的类和需要@Mock的类后,我们就可以在测试方法中对mock做文章了。
测试方法中最重要的操作是在执行service的方法之前的语句:
Mockito.when(userDao.findByNameAndPasswd(Mockito.anyString(), Mockito.anyString())) .thenReturn(mockUser);
这个方法定义了,当dao的某个方法被执行时,将会使用mock的数据返回,由此后面的操作将得到mock数据结果。
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
@Slf4j
public class MockitoTest {
@InjectMocks
private UserServiceImpl userService;
@Mock
private UserDao userDao;
@BeforeAll
public void setUp() throws Exception {
log.info("测试开始");
MockitoAnnotations.openMocks(this);
}
@AfterAll
public void tearDown() {
log.info("测试结束");
}
@Test
public void testServiceSignIn() {
UserDO mockUser = UserDO.builder().name("王五").password("王五的密码").build();
Mockito.when(userDao.findByNameAndPasswd(Mockito.anyString(), Mockito.anyString()))
.thenReturn(mockUser);
String result = userService.signIn("王五", "王五的密码");
log.info("mock返回结果:{}", result);
Assertions.assertEquals("王五,欢迎回来", result);
}
}
测试结果
INFO com.flamingo.junit5test.MockitoTest - 测试开始
INFO com.flamingo.junit5test.MockitoTest - mock返回结果:王五,欢迎回来
INFO com.flamingo.junit5test.MockitoTest - 测试结束
2.对Controller的测试,对Service的Mock
对controller的测试中,使用mockmvc发送的HTTP请求测试,因为Mockito没有接管到其中的service执行,所以还是走的真实的service方法。
而使用controller类直接调用方法时,service使用的就是Mockito生成的mock对象,因此可以返回期望的mock数据
@Autowired
private MockMvc mockMvc;
@InjectMocks
private UserController userController;
@Mock
private UserService userService;
private static final ObjectMapper MAPPER = new ObjectMapper();
@BeforeAll
public void setUp() throws Exception {
log.info("测试开始");
MockitoAnnotations.openMocks(this);
}
@Test
public void testControllerSignIn() throws Exception{
UserDO mockUser = UserDO.builder().name("赵六").password("赵六的密码").build();
Mockito.when(userService.signIn(Mockito.anyString(), Mockito.anyString()))
.thenReturn("赵六,欢迎回来");
RequestBuilder requestBuilder = MockMvcRequestBuilders.post("/user/signIn")
.contentType(MediaType.APPLICATION_JSON)
.content(MAPPER.writeValueAsString(mockUser));
MvcResult mvcResult = mockMvc.perform(requestBuilder)
.andExpect(MockMvcResultMatchers.status().isOk())
.andDo(MockMvcResultHandlers.print())
.andReturn();
log.info("【用户登录】{}", mvcResult.getResponse().getContentAsString());
String result = userController.signIn(mockUser);
log.info("【Mock用户登录】{}", result);
}
测试结果
UserControllerTest : 【用户登录】用户名或密码不正确
UserControllerTest : 【Mock用户登录】赵六,欢迎回来
四、总结及更多
? Mockito除了上面示例中的mock Dao 和 Service之外,还可以mock对象、数据等。简而言之是万物皆可Mock。另外Mockito还有涉及很多其它一些方法及特性,如验证行为、参数匹配、重复执行等,可另行查阅资料进行学习了解。
|