作者:百转春秋
链接:https://www.jianshu.com/p/813fd69aabee
来源:简书
Spring Boot提供了一些注解和工具去帮助开发者测试他们的应用。相较于SpringBoot1.3,SpringBoot1.4对测试有了大的改进,以下示例适用SpringBoot1.4.1以及以上版本。在项目中使用Spring Boot Test支持,只需要在pom.xml引入如下配置即可:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
public int example(int x, int y, int z){
if (x>1 && z>2){
x = x + y;
}
if (y == 3 || x > 5){
x = x - 2;
}
return x;
}
1
1. spring-boot-start-test
SpringBoot提供了spring-boot-start-test
启动器,该启动器提供了常见的单元测试库:
JUnit:一个Java语言的单元测试框架
Spring Test & Spring Boot Test:为Spring Boot应用提供集成测试和工具支持
AssertJ:支持流式断言的Java测试框架
Hamcrest:一个匹配器库
Mockito:一个java mock框架
JSONassert:一个针对JSON的断言库
JsonPath:JSON XPath库
2. 常用注解
这里介绍一些Spring Boot单元测试常用的注解,更多详细请到Spring Boot官网[查看]
(http://docs.spring.io/spring-boot/docs/1.4.1.RELEASE/reference/htmlsingle/#boot-features-testing)。
@RunWith(SpringRunner.class)
JUnit运行使用Spring的测试支持。SpringRunner是SpringJUnit4ClassRunner的新名字,这样做的目的
仅仅是为了让名字看起来更简单一点。@SpringBootTest
该注解为SpringApplication创建上下文并支持Spring Boot特性,其
webEnvironment
提供如下配置:Mock
-加载WebApplicationContext并提供Mock Servlet环境,嵌入的Servlet容器不会被启动。RANDOM_PORT
-加载一个EmbeddedWebApplicationContext并提供一个真实的servlet环境。嵌入的Servlet容器将被启动并在一个随机端口上监听。DEFINED_PORT
-加载一个EmbeddedWebApplicationContext并提供一个真实的servlet环境。嵌入的Servlet容器将被启动并在一个默认的端口上监听
(application.properties配置端口或者默认端口8080)。NONE
-使用SpringApplication加载一个ApplicationContext,但是不提供任何的servlet环境。@MockBean
在你的ApplicationContext里为一个bean定义一个Mockito mock。
@SpyBean
定制化Mock某些方法。使用
@SpyBean
除了被打过桩的函数,其它的函数都将真实返回。@WebMvcTest
该注解被限制为一个单一的controller,需要利用@MockBean去Mock合作者(如service)。
1. 测试用例设计方法
根据目前现状,单元测试主要用来进行程序核心逻辑测试。逻辑覆盖测试是通过对程序逻辑结构的遍历来实现程序逻辑覆盖。从对源代码的覆盖程度不同分为以下六种标准,本文只对其中的五种进行分析(路径覆盖除外),下面从一段代码开始。
public int example(int x, int y, int z){
if (x>1 && z>2){
x = x + y;
}
if (y == 3 || x > 5){
x = x - 2;
}
return x;
}
public int example(int x, int y, int z){
if (x>1 && z>2){
x = x + y;
}
if (y == 3 || x > 5){
x = x - 2;
}
return x;
}
1
一般单元测试不会根据代码来写用例,而是会根据流程图来编写测试用例,以上代码画出的流程图如下:
函数流程图
语句覆盖
1. 概念
设计足够多的测试用例,使得被测试程序中的每条可执行语句至少被执行一次。
2. 测试用例
| 数据 | 执行路径 |
|: ------------------ |:--------------------------:|
| {x=6;y=3;z=3} | a->c->b->d->e->f |
3. 测试的充分性
假设语句`x1&&z>2`中的`&&`写成了`||`上面的测试用例是检查不出来的。
判定覆盖
1. 概念
设计足够的测试用例使得代码中的判断
真
、假
分支至少被执行一次。我们标记x>1&&z>2
为P1y==3 || x>5
为P2。
2. 测试用例
| 数据 | P1 | P2 | 执行路径 |
|: ------------------ |:--------------------------:|:------------------:|:------------------:|
| {x=3;y=3;z=3} |T| T | a->c->b->d->e->f |
| {x=0;y=2;z=3} |F|F| a->c->d->f |
3. 测试的充分性
假设语句`y==3 || x>5`中的`||`写成了`&&`上面的测试用例是检查不出来的。
和语句覆盖相比:由于判定覆盖不是在判断假分支就是在判断真分支,所以满足了判定覆盖就一定会满足语句覆盖。
条件覆盖
1. 概念
设计足够多的测试用例,使得被测试程序每个判断语句中的每个逻辑条件的可能值至少满足一次。在本例中有两个判断分支`(x>1&&z>2)`和` (y == 3 || x > 5)`分别记为P1和P2。总共有三个条件`x>1`、`z>2`、`y==3`和`x>5`分别记为B1、B2、B3、B4。
2. 测试用例
数据 | P1 | P2 | B1 | B2 | B3 | B4 | 执行路径 |
---|---|---|---|---|---|---|---|
{x=0;y=2;z=3} | F | F | F | T | F | F | a->c->d->f |
{x=3;y=3;z=1} | F | T | T | F | T | T | a->c->d->f |
3. 测试的充分性
从上面的结论看,条件覆盖没法满足100%的语句覆盖,当然没法满足100%的判定覆盖。
判定/条件覆盖
1. 概念
同时满足100%的条件覆盖和100%的判定覆盖。
2. 测试用例
数据 | P1 | P2 | B1 | B2 | B3 | B4 | 执行路径 |
---|---|---|---|---|---|---|---|
{x=0;y=2;z=1} | F | F | F | F | F | F | a->c->d->f |
{x=3;y=3;z=3} | T | T | T | T | T | T | a->c->b->d->e->f |
3. 测试的充分性
达到100%判定-条件覆盖标准一定能够达到100%条件覆盖、100%判定覆盖和100%语
句覆盖。
条件组合覆盖
1. 概念
注意
:本例中判断
(x>1&&z>2)
有如下组合:(1)x>1&&z>2 (2)x>1&&z<=2 (3)x<=1&&z>2 (4) x<=1&&z<=2;判断(y == 3 || x > 5)
有如下组合(1)y==3||x>5 (2)y==3||x<=5 (3)y!=3||x>5 (4)y!=3||x<=5设计足够多的测试用例,使得被测试程序中的每个判断的所有可能条件取值的组合至少被满足一次。
条件组合只针对同一个判断语句内存在多个判断条件,让这些条件的取值进行笛卡尔乘积组合。
不同判断语句内的条件无需组合
对于单条件语句,只需要满足自己的所有取值即可
2. 测试用例
条件组合 | 数据 | 执行路径 |
---|---|---|
x>1 and z>2 y==3 or x>5 | {x=3;z=3;y=3} | a->c->b->d->e->f |
x>1 and z<=2 y==3 or x<=5 | {x=2;z=3;y=3} | a->c->b->d->f |
x<=1 and z>2 y!=3 or x>5 | {x=1;z=3;y=5} | a->c->d->e->f |
x<=1 and z<=2 y!=3 or x<=5 | {x=1;z=2;y=3} | a->c->d->f |
3. 测试的充分性
100%满足条件组合标准一定满足100%条件覆盖标准和100%判定覆盖标准。
public int example(int x, int y, int z){
if (x>1 && z>2){
x = x + y;
}
if (y == 3 || x > 5){
x = x - 2;
}
return x;
}
0
public int example(int x, int y, int z){
if (x>1 && z>2){
x = x + y;
}
if (y == 3 || x > 5){
x = x - 2;
}
return x;
}
1
在测试Controller时需要进行隔离测试,这个时候需要Mock Service层的服务。
public int example(int x, int y, int z){
if (x>1 && z>2){
x = x + y;
}
if (y == 3 || x > 5){
x = x - 2;
}
return x;
}
2
public int example(int x, int y, int z){
if (x>1 && z>2){
x = x + y;
}
if (y == 3 || x > 5){
x = x - 2;
}
return x;
}
1
测试Service和测试Controller类似,同样采用隔离法。
public int example(int x, int y, int z){
if (x>1 && z>2){
x = x + y;
}
if (y == 3 || x > 5){
x = x - 2;
}
return x;
}
1
public int example(int x, int y, int z){
if (x>1 && z>2){
x = x + y;
}
if (y == 3 || x > 5){
x = x - 2;
}
return x;
}
4
public int example(int x, int y, int z){
if (x>1 && z>2){
x = x + y;
}
if (y == 3 || x > 5){
x = x - 2;
}
return x;
}
1
测试的时候为了防止引入脏数据使用注解@Transactional和@Rollback在测试完成后进行回滚。
public int example(int x, int y, int z){
if (x>1 && z>2){
x = x + y;
}
if (y == 3 || x > 5){
x = x - 2;
}
return x;
}
7
本书第1章与第2章介绍软件单元测试的概念和基础知识。
第1章简单介绍软件单元测试所包含的概念,包括桩对象和测试驱动函数、测试驱动开发、软件测试贯彻始终、软件测试金字塔、单元测试在传统/敏捷开发模式中的地位、精准测试、单元测试和白盒测试,以及单元测试的FIRST原则和AIR原则。
第2章介绍软件单元测试基础知识,包括动态自动化/手工单元测试、静态自动化/手工单元测试。在动态自动化单元测试中介绍了语句覆盖、分支覆盖、条件覆盖、条件/分支覆盖、MC/DC、路径覆盖和控制流覆盖。
第3章到第5章介绍C语言、Java语言和Python语言的单元测试框架。
第3章介绍C语言动态自动化单元测试框架,包括在Windows下安装C语言运行环境、在Windows和Linux下安装编译CUnit、查看测试报告、CUnit介绍和案例。
第4章介绍Java语言动态自动化单元测试框架,包括在Eclipse中创建Maven项目和配置JUnit与TestNG运行环境、JUnit 4测试框架、JUnit 5测试框架、TestNG测试框架、测试替身、变异测试、利用EvoSuite自动生成测试用例,以及在Jenkins中配置JUnit 4、JUnit 5、TestNG和Allure。
第5章介绍Python语言动态自动化单元测试框架,包括unittest、Pytest及Python的模拟对象和变异测试工具mutpy。
第6章与第7章介绍代码覆盖率工具和代码语法规范检查工具。
第6章介绍代码覆盖率工具,包括C语言覆盖率工具gcov和lcov、Java语言覆盖率工具JaCoCo,以及Python语言覆盖率工具Coverage和pytest-cov。
第7章介绍代码语法规范检查工具,包括Java语言静态分析工具PMD、Python语言静态分析工具flake8和pylint,以及多代码语法规范检查平台SonarQube。
第8章通过两个案例详细介绍TDD。
读者可以根据自己的需求对以上内容进行选择性阅读或者全部阅读。另外,为了巩固大家的学习效果,每一章结尾都有相应的习题。
推荐站内搜索:最好用的开发软件、免费开源系统、渗透测试工具云盘下载、最新渗透测试资料、最新黑客工具下载……
还没有评论,来说两句吧...