我自己接触到的一些商业或是开源的基于 SpringBoot 项目,它们大部分是没有测试代码的,test 文件夹只有脚手架初始化生成的那个测试类,跟不同的开发聊到这个话题,发现他们中的大部分没有写测试的习惯,或者是觉得写测试代码麻烦,主要还是依赖测试工程师做黑盒的测试。只做黑盒测试的话有一定的的局限性,一些比较***钻的案例可能就覆盖不到,而且相对来说人也比较容易出错、遗漏。而测试代码能解决其中很大一部分的问题,利用好单元测试和集成测试在某些情况下相对于直接通过 UI 进行测试是要更方便、节省时间的,所以想通过几篇博客来分享一下自己的测试实践

为什么要写测试(优点)

  1. 覆盖更多的边界条件,且随时都可以运行测试代码(一劳永逸)
  2. 缩小测试范围:测试某个方法只需要运行对应的测试代码,而不需要运行整个项目通过请求接口进行测试
  3. 对重构更友好,可以随时重构有集成测试的代码,不用担心打破原有的代码
  4. 其他人也可以通过测试快速地理清楚对应被测代码的主线逻辑(类似文档的作用,特别是复杂代码,通过测试能快速理解上手)
  5. 写测试的过程,给自己一个新的视角去审视代码结构的设计,有助于改善代码设计

当然代码方式的测试也并非完美无缺:测试代码增加编写和维护的成本,同时一些外部依赖也需要通过 Mock 的方式实现,这些都提高了整个测试编写的门槛。也倒逼我们思考更好地组织代码,减少依赖

另一个方面:测试对于重构也是至关重要的,随着对业务的理解越来越深刻,可以重构代码,抽象出了一些共性的逻辑,优化代码结构,但是如果没有相关测试,面对着旧代码就只能望而却步了

测试工具:JUnit 5, AssertJ,Mockito

spring-boot-starter-test 自带常用的测试工具:JUnit5AssertjMockito,可以直接使用

JUnit5

Junit 5 包含:

  • JUnit Platform:Test Engine
  • Jupiter:编程模型和拓展模型
  • Vintage:兼容老版本

JUnit 4 和 5 使用的包有所不同

// JUnit 4
import org.junit.Test;
import static org.junit.Assert.assertEquals; // JUnit 5
import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.assertEquals;

如果不考虑兼容 JUnit 4 的测试,我们可以直接在依赖中直接排除 JUnit 4 的依赖,这样也可以避免在使用的时候错误地引入 JUnit 4 的包

<dependencies>

    <dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<version>{spring-boot-starter-test-version}</version>
<scope>test</scope>
<exclusions>
<exclusion>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
</exclusion>
<exclusion>
<groupId>org.junit.vintage</groupId>
<artifactId>junit-vintage-engine</artifactId>
</exclusion>
</exclusions>
</dependency> </dependencies>

还有一点值得注意的是:JUnit 5 中 @RunWith 被 @ExtendWith 代替

AssertJ

AssertJ 是一种支持链式编程的断言库,相对于 JUnit 自带的断言,它提供了更多的方法,也提供了更好的断言不匹配时的信息展示

import org.junit.jupiter.api.Test;
import static org.assertj.core.api.Assertions.assertThat; public class MyTest {
// 变量断言
@Test
public void test() {
String name = "Alice";
int age = 30; assertThat(name).isEqualTo("Alice");
assertThat(age).isGreaterThan(18).isLessThan(60);
} // List 断言
@Test
public void testList() {
List<String> list = Arrays.asList("foo", "bar", "baz")
assertThat(list).containsExactly("foo", "bar", "baz").hasSize(3);
} // Map 断言
@Test
public void testMap() {
Map<String, Integer> map = new HashMap<>();
map.put("apple", 1);
map.put("banana", 2);
map.put("orange", 3); assertThat(map).containsEntry("banana", 2); assertThat(map).containsKey("banana"); assertThat(map).containsValue(2);
} // hasNoNullFieldsOrProperties 来断言测试对象的每个属性都不为 null
@Test
public void testHasNoNullFieldsOrProperties() {
Person person = new Person("Alice", 30);
assertThat(person).hasNoNullFieldsOrProperties();
} // 异常断言
@Test
public void testDivideByZeroThrowsException() {
assertThatThrownBy(() -> {
int result = 1 / 0;
}).isInstanceOf(ArithmeticException.class)
.hasMessageContaining("/ by zero");
}
}

Mockito

Mockito 是一个 Java Mock 框架,用于创建各种类型的 Mock 对象,并设置 Mock 对象的行为

import static org.mockito.Mockito.*;

import org.junit.jupiter.api.Test;

public class MyTest {
// 创建 mock 对象
@Test
public void testCreateMock() {
List<String> list = mock(List.class);
} // 设置 mock 对象的行为
@Test
public void testMockBehavior() {
List<String> list = mock(List.class);
when(list.get(0)).thenReturn("foo");
when(list.size()).thenReturn(1);
} // 验证 mock 对象的方法调用
@Test
public void testMockVerification() {
List<String> list = mock(List.class);
list.add("foo");
list.add("bar");
verify(list).add("foo");
verify(list).add("bar");
// 验证调用方法的次数
verify(list, times(2)).add(anyString());
} // 模拟方法抛出异常
@Test
public void testMockException() {
List<String> list = mock(List.class);
doThrow(new RuntimeException()).when(list).clear();
assertThrows(RuntimeException.class, () -> list.clear());
}
}

也可以用注解来声明 Mock 对象,这样更清晰

import static org.mockito.Mockito.*;

import java.util.List;

import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.extension.ExtendWith;
import org.mockito.InjectMocks;
import org.mockito.Mock;
import org.mockito.junit.jupiter.MockitoExtension; // 启用 Mockito 扩展
@ExtendWith(MockitoExtension.class)
public class MyMockitoTest { // 创建一个 List 类型的 Mock 对象
@Mock
List<String> mockList; // 使用 @InjectMocks,将会自动注入被测试类中所声明的 Mock 对象到这个对象中
@InjectMocks
MyService myService; @Test
public void testMock() {
// 模拟 Mock 对象的行为
when(mockList.get(0)).thenReturn("foo");
when(mockList.size()).thenReturn(1); String result = myService.doSomething(); assertEquals("foo", result); verify(mockList).get(0);
}
} class MyService { private List<String> list; public MyService(List<String> list) {
this.list = list;
} public String doSomething() {
return list.get(0);
}
}

参考资料

《重构 改善既有代码的设计》

有哪个开源项目,单元测试用例覆盖的比较全的?

业务代码写单元测试的最佳姿势是什么?

单元测试有落地效果好的团队吗?

Modern Best Practices for Testing in Java

Thoughts on efficient enterprise testing (1/6) - Sebastian Daschner

SpringBoot 测试实践 - 1:常用的工具的更多相关文章

  1. Docker与自动化测试及其测试实践

    Docker 与自动化测试 对于重复枯燥的手动测试任务,可以考虑将其进行自动化改造.自动化的成本在于自动化程序的编写和维护,而收益在于节省了手动执行用例的时间.简而言之,如果收益大于成本,测试任务就有 ...

  2. Golang项目的测试实践

    Golang项目的测试实践 最近有一个项目,链路涉及了4个服务.最核心的是一个配时服务.要如何对这个项目进行测试,保证输出质量,是最近思考和实践的重点.这篇就说下最近这个实践的过程总结. 测试金字塔 ...

  3. 9-MySQL DBA笔记-测试实践

    第9章 测试实践 在第8章中介绍了测试所需要的理论知识,本章将为读者讲述实际的测试过程.实际测试一般包括硬件测试.MySQL基准测试及应用服务压力测试,下面将分别讲述这三方面的内容.此外,测试工具的选 ...

  4. 携程酒店DevOps测试实践

    作者简介 王幸福,携程酒店研发部高级测试经理,负责无线自动化测试相关工作.在测试框架和平台研发.移动测试.DevOps等领域有着丰富的经验. 如今很多大型互联网公司.创新型企业都在积极地进行DevOp ...

  5. SharePoint 2013常用开发工具分享

    众所周知,一款好的开发工具不仅能提高项目开发效率,而且能够协助开发人员简化开发流程.本文汇总几款SharePoint 2013开发常用开发工具,希望能够对大家有所帮助.如果您有更好的工具,没有包含在本 ...

  6. 常用自动化测试工具介绍(支持B/S、C/S)

    一.功能测试工具1.QTP测试工具 全名HP QuickTest Professional software ,最新的版本为HP QuickTest Professional 11.0 QTP是qui ...

  7. Drupal常用开发工具(一)——Devel模块

    进行 Drupal 开发时有许多模块和工具可供使用,其中最常用的两项便是 Devel 及 Drupal for Firebug.本文和<Drupal常用开发工具(二)——Drupal for F ...

  8. SharePoint 2013常用开发工具

    SharePoint 2013常用开发工具分享 2014-04-01 00:59 by jv9, 589 阅读, 1 评论, 收藏, 编辑 众所周知,一款好的开发工具不仅能提高项目开发效率,而且能够协 ...

  9. springboot测试、打包、部署

    本文使用<springboot集成mybatis(一)>项目,依次介绍springboot测试.打包.部署. 大多数朋友是做后端的,也就是为其他系统或者前端UI提供Rest API服务. ...

  10. Linux常用网络工具:路由扫描之mtr

    除了上一篇<Linux常用网络工具:路由扫描之traceroute>介绍的traceroute之外,一般Linux还内置了另一个常用的路由扫描工具mtr. mtr在某些方面比tracero ...

随机推荐

  1. Solon v2.2.17 发布,Java 新的生态型应用开发框架

    相对于 Spring Boot 和 Spring Cloud 的项目: 启动快 5 - 10 倍. (更快) qps 高 2- 3 倍. (更高) 运行时内存节省 1/3 ~ 1/2. (更少) 打包 ...

  2. C语言访问数据对象在内存中真实位模式的一个方法

    在判定机器采用大端还是小端存储时,可以按字节输出某数据对象的机器表示的位模式.机器表示的位模式即某数据对象在内存中的二进制串.下面是一个访问数据对象位模式的方法: //传入一个数据对象,从低地址到高地 ...

  3. js 获取系统yyyyMMdd时间

    var myDate = new Date(); var Time1 = myDate.toLocaleDateString()//yyyy/MM/dd 这个方法如果是1月份,会显示yyyy/M/dd ...

  4. Serverless云上作战阵型 | 通过云函数使用云数据库快速突破音障

    随着航空塔台的指令在耳边响起,飞行员奔向此次作战行动的两架座机.雷厉风行的爬进驾驶舱,关上舱盖,迅速下载简化操作的Demo包到机载电脑,从容的打开发动机,驾驶战斗机缓缓滑入跑道,后面僚机也已准备就绪. ...

  5. 洛谷 P8026 [ONTAK2015] Bajtocja

    简要题意 有 \(d\) 张初始为空的无向图,每张中都有 \(n\) 个点,标号从 \(1\) 到 \(n\),\(m\) 次操作,每次往一张图加一条边,并询问有多少有序数对 \((a, b)\) 使 ...

  6. Java 输入字符串,统计大写字母,小写字母,数字字符的个数

    代码如下: public static void main(String[] args) { String str = "AaFsECvcS12483fs+-*/"; int bi ...

  7. 国标GB28181协议客户端开发(三)查询和实时视频画面

    国标GB28181协议客户端开发(三)查询和实时视频画面 本文是<国标GB28181协议设备端开发>系列的第三篇,探讨了信息查询和实时视频在GB28181协议中的应用.首先,介绍了设备目录 ...

  8. LSP协议被劫持,导致无法上网

    QQ无法登录,网页打不开 用火绒的断网修复 说已经修复了 结果屁用没有 然后找的百度经验 管理员打开命令行窗口 输入 netsh winsock reset catalog 重启即生效

  9. 电脑安装JDk

    JDK软件下载链接:https://pan.baidu.com/s/1OG6wD-Fvgxu6FwuOUMDmQQ提取码:yu0l Eclipse软件下载链接:https://pan.baidu.co ...

  10. 常用语言的线程模型(Java、go、C++、python3)

    背景知识 软件是如何驱动硬件的? 硬件是需要相关的驱动程序才能执行,而驱动程序是安装在操作系统内核中.如果写了一个程序A,A程序想操作硬件工作,首先需要进行系统调用,由内核去找对应的驱动程序驱使硬件工 ...