SpringBoot单元测试实例

  • 测试实例
  • 1、创建基于Spring Data JPA的Web应用ch9_2
  • 2、由于我springBoot是2.4.4,需要额外引入JUnit
  • 3、配置数据库连接等基本属性
  • 4、创建持久化实体类
  • 5、创建数据访问层
  • 6、创建控制器层
  • 7、创建测试用例
  • 7.1、创建基于@WebMvcTest的测试用例
  • 7.2、创建基于@SpringBootTest的测试用例
  • 8、运行
  • 8.1 运行WebMvcTestStudentController测试类,执行结果如下:
  • 8.2、执行SpringBootTestStudentController
  • 9、总结
  • 9.1、@SpringBootTest和@WebMvcTest的区别是什么?

测试实例

这里我们分别使用@WebMvcTest和@SpringBootTest两种方式测试一个控制器方法是否满足测试用例。

1、创建基于Spring Data JPA的Web应用ch9_2

springboot 单元测试 null springboot 单元测试 连接数据库_spring


springboot 单元测试 null springboot 单元测试 连接数据库_JSON_02

2、由于我springBoot是2.4.4,需要额外引入JUnit

在pom.xml文件中添加

<dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
        </dependency>

3、配置数据库连接等基本属性

修改配置文件application.properties如下:

server.servlet.context-path=/ch9_2
###
##数据源信息配置
###
#数据库地址
spring.datasource.url=jdbc:mysql://localhost:3306/springbootjpa?characterEncoding=utf8
#数据库用户名
spring.datasource.username=root
#数据库密码
spring.datasource.password=123456
#数据库驱动
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
####
#JPA持久化配置
####
#指定数据库类型
spring.jpa.database=MYSQL
#指定是否在日志中显示SQL语句
spring.jpa.show-sql=true
#指定自动创建、更新数据库表等配置,update表示如果数据库中存在持久化类对应的表就不创建,不存在就创建对应的表
spring.jpa.hibernate.ddl-auto=update
#让控制器输出的JSON字符串格式更美观
spring.jackson.serialization.indent-output=true

4、创建持久化实体类

创建名为com.ch.ch9_2.entity的包,并添加Student实体类

package com.ch.ch9_2.entity;
import java.io.Serializable;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.Table;
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
@Entity
@Table(name = "student_table")
/**解决No serializer found for class org.hibernate.proxy.pojo.bytebuddy.ByteBuddyInterceptor异常*/
@JsonIgnoreProperties(value = {"hibernateLazyInitializer"})
public class Student implements Serializable{
	private static final long serialVersionUID = 1L;
	@Id
	@GeneratedValue(strategy = GenerationType.IDENTITY)
	private int id;//主键
	private String sno;
	private String sname;
	private String ssex;
	public Student() {
		super();
	}
	public Student(int id, String sno, String sname, String ssex) {
		super();
		this.id = id;
		this.sno = sno;
		this.sname = sname;
		this.ssex = ssex;
	}
	public int getId() {
		return id;
	}
	public void setId(int id) {
		this.id = id;
	}
	public String getSno() {
		return sno;
	}
	public void setSno(String sno) {
		this.sno = sno;
	}
	public String getSname() {
		return sname;
	}
	public void setSname(String sname) {
		this.sname = sname;
	}
	public String getSsex() {
		return ssex;
	}
	public void setSsex(String ssex) {
		this.ssex = ssex;
	}
}

5、创建数据访问层

创建名为com.ch.ch9_2.repository的包,并在改包中创建数据访问接口StudentRepository

package com.ch.ch9_2.repository;
import org.springframework.data.jpa.repository.JpaRepository;
import com.ch.ch9_2.entity.Student;
public interface StudentRepository extends JpaRepository<Student, Integer>{
	
}

6、创建控制器层

创建名为com.ch.ch9_2.controller的包,并添加控制器类StudentController

package com.ch.ch9_2.controller;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import com.ch.ch9_2.entity.Student;
import com.ch.ch9_2.repository.StudentRepository;
@RestController
@RequestMapping("/student")
public class StudentController {
	@Autowired
	private StudentRepository studentRepository;
	/**
	 * 保存学生信息
	 */
	@PostMapping("/save")
	public String save(@RequestBody Student student) {
		studentRepository.save(student);
		return "success";
	}
	/**
	 * 根据id查询学生信息
	 */
	@GetMapping("/getOne/{id}")
	public Student getOne(@PathVariable("id") int id){
		return studentRepository.getOne(id);
	}
}

7、创建测试用例

7.1、创建基于@WebMvcTest的测试用例

使用@WebMvcTest注解测试Controller时,带有@Service以及别的注解组件类不会自动被扫描注册为Spring容器管理的Bean。因此,Controller所依赖的对象必须用@MockBean来模拟实现。
在src/test/java目录下的com.ch.ch9_2包中,创建基于@WebMvcTest的测试用例类WebMvcTestStudentController

package com.ch.ch9_2;
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get;
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post;
import static org.springframework.test.web.servlet.result.MockMvcResultHandlers.print;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.BDDMockito;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest;
import org.springframework.boot.test.mock.mockito.MockBean;
import org.springframework.http.MediaType;
import org.springframework.test.context.junit4.SpringRunner;
import org.springframework.test.web.servlet.MockMvc;
import com.ch.ch9_2.controller.StudentController;
import com.ch.ch9_2.entity.Student;
import com.ch.ch9_2.repository.StudentRepository;
import com.fasterxml.jackson.databind.ObjectMapper;
@RunWith(SpringRunner.class)
/*仅仅扫描这个StudentController类,即注入StudentController到Spring容器*/
@WebMvcTest(StudentController.class)
public class WebMvcTestStudentController {
	//MockMvc是Spring提供的专用于测试Controller类
	@Autowired
	private MockMvc mvc;
	//因为在StudentController类依赖StudentRepository,所以需要mock掉
	@MockBean
	private StudentRepository studentRepository; 
	@Test
	public void saveTest() throws Exception {
		Student stu = new Student(1,"5555","陈恒","男");
		ObjectMapper mapper = new ObjectMapper();//把对象转换成JSON字符串
		mvc.perform(post("/student/save")
				.contentType(MediaType.APPLICATION_JSON_UTF8)//发送JSON数据格式
				.accept(MediaType.APPLICATION_JSON_UTF8)//接收JSON数据格式
				.content(mapper.writeValueAsString(stu))//传递JSON字符串参数
				)
		.andExpect(status().isOk())//状态响应码为200,如果不是抛出异常,测试不通过。
		.andDo(print());//输出结果
	}
	@Test
	public void getStudent() throws Exception {
		Student stu = new Student(1,"5555","陈恒","男");
		//模拟StudentRepository,getOne(1)将返回stu对象
		BDDMockito.given(studentRepository.getOne(1)).willReturn(stu);
		mvc.perform(get("/student/getOne/{id}", 1)
				.contentType(MediaType.APPLICATION_JSON_UTF8)
				.accept(MediaType.APPLICATION_JSON_UTF8)
				)
				.andExpect(status().isOk())//状态响应码为200,如果不是抛出异常,测试不通过。
				.andExpect(jsonPath("$.sname").value("陈恒"))
				.andDo(print());//输出结果
	}
}

7.2、创建基于@SpringBootTest的测试用例

@SpringBootTest注解告诉Spring Boot去寻找一个主配置类(一个带@SpringBootApplication的类),并使用它启动Spring应用程序的上下文,同时注入所有Bean。
在src/test/java目录下的com.ch.ch9_2包中,创建基于@SpringBootTest的测试用例类SpringBootTestStudentController

package com.ch.ch9_2;
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get;
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post;
import static org.springframework.test.web.servlet.result.MockMvcResultHandlers.print;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.http.MediaType;
import org.springframework.test.context.junit4.SpringRunner;
import org.springframework.test.web.servlet.MockMvc;
import org.springframework.test.web.servlet.setup.MockMvcBuilders;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.web.context.WebApplicationContext;
import com.ch.ch9_2.entity.Student;
import com.fasterxml.jackson.databind.ObjectMapper;
@RunWith(SpringRunner.class)
@SpringBootTest(classes = Ch92Application.class)//应用的主程序
public class SpringBootTestStudentController {
	//注入Spring容器
	@Autowired
	private WebApplicationContext wac;
	//MockMvc模拟实现对Controller的请求
	private MockMvc mvc;
	//在测试前,初始化MockMvc对象
	@Before
	public void initMockMvc() {
		mvc = MockMvcBuilders.webAppContextSetup(wac).build();
	}
	@Test
	@Transactional
	public void saveTest() throws Exception {
		Student stu = new Student(1, "5555","陈恒","男");
		ObjectMapper mapper = new ObjectMapper();//把对象转换成JSON字符串
		mvc.perform(post("/student/save")
				.contentType(MediaType.APPLICATION_JSON_UTF8)//发送JSON数据格式
				.accept(MediaType.APPLICATION_JSON_UTF8)//接收JSON数据格式
				.content(mapper.writeValueAsString(stu))//传递JSON字符串参数
				)
		.andExpect(status().isOk())//状态响应码为200,如果不是抛出异常,测试不通过。
		.andDo(print());//输出结果
	}
	@Test
	public void getStudent() throws Exception {
		mvc.perform(get("/student/getOne/{id}", 1)
				.contentType(MediaType.APPLICATION_JSON_UTF8)
				.accept(MediaType.APPLICATION_JSON_UTF8)
				)
				.andExpect(status().isOk())//状态响应码为200,如果不是抛出异常,测试不通过。
				.andExpect(jsonPath("$.sname").value("陈恒"))
				.andDo(print());//输出结果
	}
}

8、运行

8.1 运行WebMvcTestStudentController测试类,执行结果如下:

MockHttpServletRequest:
      HTTP Method = POST
      Request URI = /student/save
       Parameters = {}
          Headers = [Content-Type:"application/json;charset=UTF-8", Accept:"application/json;charset=UTF-8", Content-Length:"51"]
             Body = {"id":1,"sno":"5555","sname":"陈恒","ssex":"男"}
    Session Attrs = {}

Handler:
             Type = com.ch.ch9_2.controller.StudentController
           Method = com.ch.ch9_2.controller.StudentController#save(Student)

Async:
    Async started = false
     Async result = null

Resolved Exception:
             Type = null

ModelAndView:
        View name = null
             View = null
            Model = null

FlashMap:
       Attributes = null

MockHttpServletResponse:
           Status = 200
    Error message = null
          Headers = [Content-Type:"application/json;charset=UTF-8", Content-Length:"7"]
     Content type = application/json;charset=UTF-8
             Body = success
    Forwarded URL = null
   Redirected URL = null
          Cookies = []

MockHttpServletRequest:
      HTTP Method = GET
      Request URI = /student/getOne/1
       Parameters = {}
          Headers = [Content-Type:"application/json;charset=UTF-8", Accept:"application/json;charset=UTF-8"]
             Body = null
    Session Attrs = {}

Handler:
             Type = com.ch.ch9_2.controller.StudentController
           Method = com.ch.ch9_2.controller.StudentController#getOne(int)

Async:
    Async started = false
     Async result = null

Resolved Exception:
             Type = null

ModelAndView:
        View name = null
             View = null
            Model = null

FlashMap:
       Attributes = null

MockHttpServletResponse:
           Status = 200
    Error message = null
          Headers = [Content-Type:"application/json;charset=UTF-8"]
     Content type = application/json;charset=UTF-8
             Body = {
  "id" : 1,
  "sno" : "5555",
  "sname" : "陈恒",
  "ssex" : "男"
}
    Forwarded URL = null
   Redirected URL = null
          Cookies = []
2021-04-09 21:30:14.183  INFO 23076 --- [extShutdownHook] o.s.s.concurrent.ThreadPoolTaskExecutor  : Shutting down ExecutorService 'applicationTaskExecutor'

Process finished with exit code 0

状态码为200说明执行成功了

8.2、执行SpringBootTestStudentController

注意,这次是查了数据库的,和上面的不一样,上面的例子是造了个数据,这个是真的查了数据库之后比较的
我们的数据是

“id” : 1,
“sno” : “5555”,
“sname” : “陈恒”,
“ssex” : “男”

但是数据库里面的数据是:

springboot 单元测试 null springboot 单元测试 连接数据库_spring_03

Hibernate: select student0_.id as id1_0_0_, student0_.sname as sname2_0_0_, student0_.sno as sno3_0_0_, student0_.ssex as ssex4_0_0_ from student_table student0_ where student0_.id=?

MockHttpServletRequest:
      HTTP Method = POST
      Request URI = /student/save
       Parameters = {}
          Headers = [Content-Type:"application/json;charset=UTF-8", Accept:"application/json;charset=UTF-8", Content-Length:"51"]
             Body = {"id":1,"sno":"5555","sname":"陈恒","ssex":"男"}
    Session Attrs = {}

Handler:
             Type = com.ch.ch9_2.controller.StudentController
           Method = com.ch.ch9_2.controller.StudentController#save(Student)

Async:
    Async started = false
     Async result = null

Resolved Exception:
             Type = null

ModelAndView:
        View name = null
             View = null
            Model = null

FlashMap:
       Attributes = null

MockHttpServletResponse:
           Status = 200
    Error message = null
          Headers = [Content-Type:"application/json;charset=UTF-8", Content-Length:"7"]
     Content type = application/json;charset=UTF-8
             Body = success
    Forwarded URL = null
   Redirected URL = null
          Cookies = []
2021-04-09 21:32:14.765  INFO 24340 --- [           main] o.s.t.c.transaction.TransactionContext   : Rolled back transaction for test: [DefaultTestContext@327514f testClass = SpringBootTestStudentController, testInstance = com.ch.ch9_2.SpringBootTestStudentController@5a0f87e5, testMethod = saveTest@SpringBootTestStudentController, testException = [null], mergedContextConfiguration = [WebMergedContextConfiguration@5b12b668 testClass = SpringBootTestStudentController, locations = '{}', classes = '{class com.ch.ch9_2.Ch92Application}', contextInitializerClasses = '[]', activeProfiles = '{}', propertySourceLocations = '{}', propertySourceProperties = '{org.springframework.boot.test.context.SpringBootTestContextBootstrapper=true}', contextCustomizers = set[org.springframework.boot.test.context.filter.ExcludeFilterContextCustomizer@a74868d, org.springframework.boot.test.json.DuplicateJsonObjectContextCustomizerFactory$DuplicateJsonObjectContextCustomizer@365185bd, org.springframework.boot.test.mock.mockito.MockitoContextCustomizer@0, org.springframework.boot.test.web.client.TestRestTemplateContextCustomizer@4cf4d528, org.springframework.boot.test.autoconfigure.actuate.metrics.MetricsExportContextCustomizerFactory$DisableMetricExportContextCustomizer@df27fae, org.springframework.boot.test.autoconfigure.properties.PropertyMappingContextCustomizer@0, org.springframework.boot.test.autoconfigure.web.servlet.WebDriverContextCustomizerFactory$Customizer@587c290d, org.springframework.boot.test.context.SpringBootTestArgs@1, org.springframework.boot.test.context.SpringBootTestWebEnvironment@5e853265], resourceBasePath = 'src/main/webapp', contextLoader = 'org.springframework.boot.test.context.SpringBootContextLoader', parent = [null]], attributes = map['org.springframework.test.context.web.ServletTestExecutionListener.activateListener' -> true, 'org.springframework.test.context.web.ServletTestExecutionListener.populatedRequestContextHolder' -> true, 'org.springframework.test.context.web.ServletTestExecutionListener.resetRequestContextHolder' -> true, 'org.springframework.test.context.event.ApplicationEventsTestExecutionListener.recordApplicationEvents' -> false]]
2021-04-09 21:32:14.770  INFO 24340 --- [           main] o.s.b.t.m.w.SpringBootMockServletContext : Initializing Spring TestDispatcherServlet ''
2021-04-09 21:32:14.770  INFO 24340 --- [           main] o.s.t.web.servlet.TestDispatcherServlet  : Initializing Servlet ''
2021-04-09 21:32:14.771  INFO 24340 --- [           main] o.s.t.web.servlet.TestDispatcherServlet  : Completed initialization in 0 ms
Hibernate: select student0_.id as id1_0_0_, student0_.sname as sname2_0_0_, student0_.sno as sno3_0_0_, student0_.ssex as ssex4_0_0_ from student_table student0_ where student0_.id=?

java.lang.AssertionError: JSON path "$.sname" 
Expected :陈恒
Actual   :张三
<Click to see difference>

点击上面的,如下图,可以发现测试成功了。查出来值不符合要求。

springboot 单元测试 null springboot 单元测试 连接数据库_spring_04

9、总结

9.1、@SpringBootTest和@WebMvcTest的区别是什么?

我们在使用@WebMvcTest注解测试Controller时,带有@Service以及别的注解组件类不会自动被扫描注册为Spring容器管理的Bean,而@SpringBootTest注解告诉Spring Boot去寻找一个主配置类(一个带@SpringBootApplication的类),并使用它来启动Spring应用程序上下文,注入所有Bean。另外,还需要注意的是,MockMvc用来在Servlet容器内对Controller进行单元测试,并未真正发起了HTTP请求调用Controller。
@WebMvcTest用于从服务器端对Controller层进行统一测试;如果需要从客户端与应用程序交互时,应该使用@SpringBootTest做集成测试。