springboot单元测试详解
⽂章⽬录
⼀.Junit 测试
当你的单元测试代码不需要⽤到 Spring Boot 功能,⽽只是⼀个简单的测试时,你可以直接编写你的 Junit 测试代码:
public class SimpleJunitTest {
@Test
public void testSayHi(){
System.out.println("Hi Junit.");
}
}
⼆.集成测试
Spring 框架提供了⼀个专门的测试模块(spring-test),⽤于应⽤程序的集成测试。 在 Spring Boot 中,你可以通过spring-boot-starter-test启动器快速开启和使⽤它。
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
1.Spring Boot 测试-测试其中的bean
当你的集成测试代码需要⽤到 Spring Boot 功能时,你可以使⽤@SpringBootTest注解。
该注解是普通的 Spring 项⽬(⾮ Spring Boot 项⽬)中编写集成测试代码所使⽤的@ContextConfiguration注解的替代品。其作⽤是⽤于确定如何装载 Spring 应⽤程序的上下⽂资源
@SpringBootTest
class MysqlDataTestApplicationTests {
@Autowired
TOrderInfoService tOrderInfoService;
@Test
public void tOrderInfoServiceSelect(){
TOrderInfo tOrderInfo = tOrderInfoService.selectByPrimaryKey(1L);
Assertions.assertThat(tOrderInfo).isNotNull();
}
}
当运⾏ Spring Boot 应⽤程序测试时,它会⾃动的从当前测试类所在的包起⼀层⼀层向上搜索,直到到⼀个@SpringBootApplication 或@SpringBootConfiguration注释类为⽌。以此来确定如何装载 Spring 应⽤程序的上下⽂资源。只要你以合理的⽅式组织你的代码,你项⽬的主配置通常是可以被发
现的。
如果搜索算法搜索不到你项⽬的主配置⽂件,将报出异常:
java.lang.IllegalStateException: Unable to find a @SpringBootConfiguration, you need to use @ContextConfiguration or
@SpringBootTest(classes=…) with your test
解决办法是,按 Spring Boot 的约定重新组织你的代码结构,或者⼿⼯指定你要装载的主配置⽂件:
@SpringBootTest(classes ={MysqlDataTestApplication.class})
class MysqlDataTestApplicationTests {
@Autowired
TOrderInfoService tOrderInfoService;
@Test
public void tOrderInfoServiceSelect(){
TOrderInfo tOrderInfo = tOrderInfoService.selectByPrimaryKey(1L);
Assertions.assertThat(tOrderInfo).isNotNull();
}
}
2.Spring Boot Web 测试- 启动tomcat
当你想启动⼀个完整的 HTTP 服务器对 Spring Boot 的 Web 应⽤编写测试代码时,可以使⽤@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)注解开启⼀个随机的可⽤端⼝。Spring Boot 针对 REST 调⽤的测试提供了⼀个TestRestTemplate 模板,它可以解析链接服务器的相对地址。
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
class MysqlDataTestApplicationTests2 {
@Autowired
private TestRestTemplate restTemplate;
@Test
public void test()throws Exception {
String response = ForObject("/query", String.class);
Assertions.assertThat(response).contains("QRC123456789012");
}
}
3.Spring Boot Web 测试- 不启动tomcat(模拟环境)
@SpringBootTest
@AutoConfigureMockMvc
class MysqlDataTestApplicationTests2 {
@Autowired
private MockMvc mockMvc;
@Test
public void test()throws Exception {
mockMvc.("/query")).andDo(MockMvcResultHandlers.print())
.andExpect(MockMvcResultMatchers.status().isOk());
springboot框架的作用
}
}
三.单元测试
上⽂中描述的两种集成测试的⽅案,相同的⼀点是都会构建整个Spring Context。这表⽰所有声明的bean,⽽不管声明的⽅式为何,都会被构建实例,并且都能被依赖。这⾥隐含的意思是从上到下整条依赖链上的代码都已实现。
Mock技术
在开发的过程中进⾏测试,⽆法满⾜上述的条件,Mock技术可以让我们屏蔽掉下层的依赖,从⽽专注于当前的测试⽬标。Mock技术的思想是,当测试⽬标的下层依赖的⾏为是可预期的,那么测试⽬标本⾝的⾏为也是可预期的,测试就是把实际的结果和测试⽬标的预期结果做⽐较,⽽Mock就是预先设定下层依赖的⾏为表现。
Mock的流程
1. 将测试⽬标的依赖对象进⾏mock,设定其预期的⾏为表现。
2. 对测试⽬标进⾏测试。
3. 检测测试结果,检查在依赖对象的预期⾏为下,测试⽬标的结果是否符合预期。
Mock的使⽤场景
1. 多⼈协作时,可以通过mock进⾏⽆等待的测试先⾏。
2. 当测试⽬标的依赖对象需要访问外部的服务,⽽外部服务不易获得时,可以通过mock来模拟服务可⽤。
3. 当在排查不容易复现的问题场景时,通过mock来模拟问题。
1.web层测试
当你想对 Spring MVC 控制器编写单元测试代码时,可以使⽤@WebMvcTest注解。它提供了⾃配置的 MockMvc,可以不需要完整启动HTTP 服务器就可以快速测试 MVC 控制器。
@WebMvcTest(TestController.class)
class TestControllerTests {
@Autowired
private MockMvc mockMvc;
@MockBean
private TOrderInfoService tOrderInfoService;
@Test
public void query()throws Exception {
TOrderInfo tOrderInfo =new TOrderInfo();
tOrderInfo.setId(1L);
tOrderInfo.setMerId("QRC123456789012");
Mockito.when(tOrderInfoService.selectByPrimaryKey(1L)).thenReturn(tOrderInfo);
mockMvc.("/query")).andDo(MockMvcResultHandlers.print())
.t().ainsString("QRC123456789012")));
}
}
使⽤@WebMvcTest注解时,只有⼀部分的 Bean 能够被扫描得到,它们分别是:
@Controller
@ControllerAdvice
@JsonComponent
Filter
WebMvcConfigurer
HandlerMethodArgumentResolver
其他常规的@Component(包括@Service、@Repository等)Bean 则不会被加载到 Spring 测试环境上下⽂中。
如果测试的 MVC 控制器中需要@ComponentBean 的参与,你可以使⽤@MockBean注解来协助完成
如果使⽤了mybatis的@MapperScan注解,则mapper也会⾃动装配,但是却么有DataSource,测试会失败,可以先注释掉
@MapperScan注解
// 连接真实的数据库,否则连接⼀个内存中的库,测试会失败
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
public class TOrderInfoMapperTests {
@Autowired
TOrderInfoMapper tOrderInfoMapper;
@Test
// 测试数据不回滚
@Rollback(false)
// 事务是只读的,如果有更新操作,⽅法将失败
// @Transactional(readOnly = true)
public void test3()throws Exception {
TOrderInfo orderInfo =new TOrderInfo();
orderInfo.setMerId("133333333");
orderInfo.setId(2L);
int  i = tOrderInfoMapper.updateByPrimaryKeySelective(orderInfo);
Assertions.assertThat(i).isEqualTo(1);
}
}
3.测试任意的bean
service层并不作为⼀种特殊的层,所以没有什么注解能表⽰“只构建service层的bean”这种概念。
这⾥将介绍另⼀种通⽤的测试场景,我要测试的是⼀个普通的bean,没有什么特殊的⾓⾊,⽐如不是担当特殊处理的controller,也不是负责持久化的dao组件,我们要测试的只是⼀个普通的bean。
上⽂中我们使⽤@SpringBootTest的默认机制,它去查@SpringBootApplication的配置,据此构建Spring的上下⽂。查看
@SpringBootTest的doc,其中有⼀句是:
Automatically searches for a @SpringBootConfiguration when nested @Configuration is not used, and no explicit classes are specified.
这表⽰我们可以通过classes属性来指定Configuration类,或者定义内嵌的Configuration类来改变默认的配置。
public class TOrderInfoServiceTest {
@Configuration
static class TOrderInfoServiceConfig {
@Bean
public TOrderInfoService cityService(){
return new TOrderInfoService();
}
}
@Autowired
private TOrderInfoService tOrderInfoService;
@MockBean
private TOrderInfoMapper tOrderInfoMapper;
@Test
public void test(){
TOrderInfo tOrderInfo =new TOrderInfo();
tOrderInfo.setId(1L);
tOrderInfo.setMerId("QRC123456789012");
Mockito.when(tOrderInfoMapper.selectByPrimaryKey(1L)).thenReturn(tOrderInfo);
TOrderInfo tOrderInfo1 = tOrderInfoService.selectByPrimaryKey(1L);
Assertions.assertThat(tOrderInfo1).isNotNull();
}
}
4.Mock操作
单元测试中,需要对测试⽬标的依赖进⾏mock,这⾥有必要对mock的细节介绍下。上⽂单元测试部分已对Mock的逻辑、流程和使⽤场景进⾏了介绍,此处专注于实践层⾯进⾏说明。
根据⽅法参数设定预期⾏为
⼀般的mock是对⽅法级别的mock,在⽅法有⼊参的情况下,⽅法的⾏为可能会跟⽅法的具体参数值有关。⽐如⼀个除法的⽅法,传⼊参数4、2得结果2,传⼊参数8、2得结果4,传⼊参数2、0得异常。
mock可以针对不同的参数值设定不同的预期,如下所⽰:

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。