如何用SpringBoot 进行测试
普通测试
假设要测试一个工具类 StringUtil(com.rxliuli.example.springboottest.util.StringUtil)
/** * 用于测试的字符串工具类 * * @author rxliuli */ public class StringUtil { /** * 判断是否为空 * * @param string 要进行判断的字符串 * @return 是否为 null 或者空字符串 */ public static boolean isEmpty(String string) { return string == null || string.isEmpty(); } /** * 判断是否为空 * * @param string 要进行判断的字符串 * @return 是否为 null 或者空字符串 */ public static boolean isNotEmpty(String string) { return !isEmpty(string); } /** * 判断是否有字符串为空 * * @param strings 要进行判断的一个或多个字符串 * @return 是否有 null 或者空字符串 */ public static boolean isAnyEmpty(String... strings) { return Arrays.stream(strings) .anyMatch(StringUtil::isEmpty); } /** * 判断字符串是否全部为空 * * @param strings 要进行判断的一个或多个字符串 * @return 是否全部为 null 或者空字符串 */ public static boolean isAllEmpty(String... strings) { return Arrays.stream(strings) .allMatch(StringUtil::isEmpty); } }
需要添加依赖 spring-boot-starter-test
以及指定 assertj-core
的最新版本
<dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> </dependencies> <dependencyManagement> <dependencies> <dependency> <groupId>org.assertj</groupId> <artifactId>assertj-core</artifactId> <version>3.9.1</version> <scope>test</scope> </dependency> </dependencies> </dependencyManagement>
这里指定 assertj-core
的版本是为了使用较新的一部分断言功能(例如属性 lambda
断言)
/** * @author rxliuli */ public class StringUtilTest { private String strNull = null; private String strEmpty = ""; private String strSome = "str"; @Test public void isEmpty() { //测试 null assertThat(StringUtil.isEmpty(strNull)) .isTrue(); //测试 empty assertThat(StringUtil.isEmpty(strEmpty)) .isTrue(); //测试 some assertThat(StringUtil.isEmpty(strSome)) .isFalse(); } @Test public void isNotEmpty() { //测试 null assertThat(StringUtil.isNotEmpty(strNull)) .isFalse(); //测试 empty assertThat(StringUtil.isNotEmpty(strEmpty)) .isFalse(); //测试 some assertThat(StringUtil.isNotEmpty(strSome)) .isTrue(); } @Test public void isAnyEmpty() { assertThat(StringUtil.isAnyEmpty(strNull, strEmpty, strSome)) .isTrue(); assertThat(StringUtil.isAnyEmpty()) .isFalse(); } @Test public void isAllEmpty() { assertThat(StringUtil.isAllEmpty(strNull, strEmpty, strSome)) .isFalse(); assertThat(StringUtil.isAnyEmpty(strNull, strEmpty)) .isTrue(); } }
这里和非 SpringBoot 测试时没什么太大的区别,唯一的一点就是引入 Jar 不同,这里虽然我们只引入了 spring-boot-starter-test
,但它本身已经帮我们引入了许多的测试相关类库了。
Dao/Service 测试
从这里开始就和标准的 Spring 不太一样了
首先,我们需要 Dao 层,这里使用 H2DB 和 SpringJDBC 做数据访问层(比较简单)。
依赖
<dependency> <groupId>com.h2database</groupId> <artifactId>h2</artifactId> <scope>runtime</scope> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-jdbc</artifactId> </dependency>
添加两个初始化脚本
数据库结构 db_schema.sql
(db/db_schema.sql
)
drop table if exists user; create table user ( id int auto_increment not null comment '编号', name varchar(20) not null comment '名字', sex boolean null comment '性别', age int null comment '年龄' );
数据库数据 db_data.sql
(db/db_data.sql
)
insert into user (id, name, sex, age) values (1, '琉璃', false, 17), (2, '月姬', false, 1000);
为 SpringBoot 配置一下数据源及初始化脚本
spring: datasource: driver-class-name: org.h2.Driver platform: h2 schema: classpath:db/db_schema.sql data: classpath:db/db_data.sql
然后是实体类与 Dao
用户实体类 User
(com.rxliuli.example.springboottest.entity.User
)
/** * @author rxliuli */ public class User implements Serializable { private Integer id; private String name; private Boolean sex; private Integer age; public User() { } public User(String name, Boolean sex, Integer age) { this.name = name; this.sex = sex; this.age = age; } public User(Integer id, String name, Boolean sex, Integer age) { this.id = id; this.name = name; this.sex = sex; this.age = age; } //getter() and setter() }
用户 Dao UserDao
(com.rxliuli.example.springboottest.dao.UserDao
)
/** * @author rxliuli */ @Repository public class UserDao { private final RowMapper<User> userRowMapper = (rs, rowNum) -> new User( rs.getInt("id"), rs.getString("name"), rs.getBoolean("sex"), rs.getInt("age") ); @Autowired private JdbcTemplate jdbcTemplate; /** * 根据 id 获取一个对象 * * @param id id * @return 根据 id 查询到的对象,如果没有查到则为 null */ public User get(Integer id) { return jdbcTemplate.queryForObject("select * from user where id = ?", userRowMapper, id); } /** * 查询全部用户 * * @return 全部用户列表 */ public List<User> listForAll() { return jdbcTemplate.query("select * from user", userRowMapper); } /** * 根据 id 删除用户 * * @param id 用户 id * @return 受影响行数 */ public int deleteById(Integer id) { return jdbcTemplate.update("delete from user where id = ?", id); } }
接下来才是正事,测试 Dao 层需要加载 Spring 容器,自动回滚以避免污染数据库。
/** * {@code @SpringBootTest} 和 {@code @RunWith(SpringRunner.class)} 是必须的,这里貌似一直有人误会需要使用 {@code @RunWith(SpringJUnit4ClassRunner.class)},但其实并不需要了 * 下面的 {@code @Transactional} 和 {@code @Rollback}则是开启事务控制以及自动回滚 * * @author rxliuli */ @SpringBootTest @RunWith(SpringRunner.class) @Transactional @Rollback public class UserDaoTest { @Autowired private UserDao userDao; @Test public void get() { int id = 1; User result = userDao.get(id); //断言 id 和 get id 相同 assertThat(result) .extracting(User::getId) .contains(id); } @Test public void listForAll() { List<User> userList = userDao.listForAll(); //断言不为空 assertThat(userList) .isNotEmpty(); } @Test public void deleteById() { int result = userDao.deleteById(1); assertThat(result) .isGreaterThan(0); } }
Web 测试
与传统的 SpringTest 一样,SpringBoot 也分为两种。
- 独立安装测试:
手动加载单个 Controller,所以测试其他 Controller 中的接口会发生异常。但测试速度上较快,所以应当优先选择。
- 集成 Web 环境测试:
将启动并且加载所有的 Controller, 所以效率上之于 BaseWebUnitTest 来说非常低下, 仅适用于集成测试多个 Controller 时使用。
独立安装测试
主要是设置需要使用的 Controller 实例,然后用获得 MockMvc 对象进行测试即可。
/** * @author rxliuli */ @SpringBootTest @RunWith(SpringRunner.class) @Transactional @Rollback public class UserControllerUnitTest { @Autowired private UserController userController; /** * 用于测试 API 的模拟请求对象 */ private MockMvc mockMvc; @Before public void before() { //模拟一个 Mvc 测试环境,获取一个 MockMvc 实例 mockMvc = MockMvcBuilders.standaloneSetup(userController) .build(); } @Test public void testGet() throws Exception { //测试能够正常获取 Integer id = 1; mockMvc.perform( //发起 get 请求 get("/user/" + id) ) //断言请求的状态是成功的(200) .andExpect(status().isOk()) //断言返回对象的 id 和请求的 id 相同 .andExpect(jsonPath("$.id").value(id)); } @Test public void listForAll() throws Exception { //测试正常获取 mockMvc.perform( //发起 post 请求 post("/user/listForAll") ) //断言请求状态 .andExpect(status().isOk()) //断言返回结果是数组 .andExpect(jsonPath("$").isArray()) //断言返回数组不是空的 .andExpect(jsonPath("$").isNotEmpty()); } }
集成 Web 环境测试
/** * @author rxliuli */ @SpringBootTest @RunWith(SpringRunner.class) @Transactional @Rollback public class UserControllerIntegratedTest { @Autowired private WebApplicationContext context; /** * 用于测试 API 的模拟请求对象 */ private MockMvc mockMvc; @Before public void before() { //这里把整个 WebApplicationContext 上下文都丢进去了,所以可以测试所有的 Controller mockMvc = MockMvcBuilders.webAppContextSetup(context) .build(); } @Test public void testGet() throws Exception { //测试能够正常获取 Integer id = 1; mockMvc.perform( //发起 get 请求 get("/user/" + id) ) //断言请求的状态是成功的(200) .andExpect(status().isOk()) //断言返回对象的 id 和请求的 id 相同 .andExpect(jsonPath("$.id").value(id)); } @Test public void listForAll() throws Exception { //测试正常获取 mockMvc.perform( //发起 post 请求 post("/user/listForAll") ) //断言请求状态 .andExpect(status().isOk()) //断言返回结果是数组 .andExpect(jsonPath("$").isArray()) //断言返回数组不是空的 .andExpect(jsonPath("$").isNotEmpty()); } }
总结
其实上面的测试类的注解感觉都差不多,我们可以将一些普遍的注解封装到基类,然后测试类只要继承基类就能得到所需要的环境,吾辈自己的测试基类在 src/test/common
下面,具体使用方法便留到下次再说吧
以上代码已全部放到 GitHub 上面,可以直接 clone 下来进行测试
到此这篇关于如何用SpringBoot 进行测试的文章就介绍到这了,更多相关SpringBoot 测试内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!