单元测试的思考与实践
1. 什么是单元测试
通常来说单元测试,是一种自动化测试,同时包含一下特性:
·验证很小的一段代码(业务意义 或者 代码逻辑 上不可再分割的单元),能够更准确的定位到问题代码的位置
· 能够快速运行(单元测试的意义,在于快速且周期性的验证原有代码的准确性),提高项目开发效率
· 以隔离的方式 (isolated manner)运行(对外部依赖通过插桩解耦,避免单元测试的复杂度,实现问题快速定位,简化单元测试的运行环境,多个单元测试可以以任何顺序甚至并行进行)
2. 为什么要单元测试
因为单元测试有如下优点:
· 能快速的回归,提高自测的效率
· 集成测试或者端到端的手工测试效率低,而且无法覆盖到更细节的逻辑分支
· 也存在功能设计超前于产品设计,通过接口维度,无法触达某些逻辑分支,需要通过单元测试来覆盖
· 功能开发人员更了解代码的实现,开发人员写出的测试用例往往能更全面的覆盖代码
· 有良好单测的代码,往往更方便重构
· 单元测试是项目代码的一部分,维护方便,当然这也依赖良好的单元测试编写习惯,合适的颗粒度
3. 如何识别有测试价值的代码?
当我们考虑给代码添加 单元测试时,需要首先考虑加入单测后能够带来的收益有多少,以及其付出的成本有多少,用最小的维护成本提供最高的价值的单元测试。
3.1 项目属性
软件本身发布更新成本比较大,如嵌入式软件,客户端程序;或者 软件的缺陷 更可能带来较大的资损,如工厂,银行内部的软件,这类软件都是需要优先考虑单元测试。
如果一个项目本身不是特别核心的项目,影响面小,迭代更新相对较容易,那么对单元测试的要求,或者说对质量的要求,也就没有那么强烈。
3.2 代码属性
3.2.1 重要的代码
· 领域层
· 基础设施代码
3.2.2 不容易被集成测试覆盖的代码
· 边界条件
· 异常条件
· 低概率场景
3.2.3 容易出现问题的代码
· 复杂的业务逻辑分支
· 状态机
· 胶水代码:负责组合多个功能,多个功能的输入具有不确定性
3.3 个人不建议的单元测试的行为
通常来说不建议在单元测试的时候,启动spring容器后,会牵扯过多的外部依赖,导致单元测试难以进行,或者成本过高。
同样,外部接口,数据库依赖,中间件依赖,都不建议在单元测试中加载,可以通过mock或者sub的方式来进行隔离。
4. 编写 Unit Test
通常按照单元测试的AAA模式来编写单元测试,分为三部分:Arrange, Act, Assert
1)Arrange
准备测试数据和测试环境,确保测试的可重复性和可预测性。这包括初始化对象、设置变量、模拟外部依赖等
2)Act
执行实际的测试操作,也就是调用需要测试的方法或函数,并获取返回值或状态。这个阶段应该仅包含单个操作,以确保测试的独立性和可维护性
3)Assert
验证测试结果是否符合预期,也就是检查实际的输出是否与预期的输出相同。如果结果不符合预期,我们需要检查测试代码和被测试代码,找出问题所在并进行修复
4)结果验证 - 对函数返回结果进行验证
5)状态验证 - 对过程中的属性值来进行验证
6)行为验证 - 对过程中会执行的动作进行验证
spock测试框架代码示例:
class OrderServiceImplTest extends Specification {OrderService orderService = new OrderServiceImpl();InventoryService inventoryService = Mock(InventoryService)OrderConverter orderConverter = Mock(OrderConverter.class)PaymentChannelClient paymentChannelClient = Mock(PaymentChannelClient)OrderMapper orderMapper = Mock(OrderMapper)def setupSpec() {} // runs once - before the first feature methoddef setup() { // runs before every feature methodorderService.setInventoryService(inventoryService)orderService.setPaymentChannelClient(paymentChannelClient)orderService.setOrderMapper(orderMapper)orderService.setOrderConverter(orderConverter)}def cleanup() {} // runs after every feature methoddef cleanupSpec() {} // runs once - after the last feature methoddef "create order correctly"() {//准备测试需要的参数given:Long id = 1CreateOrderCommand command = new CreateOrderCommand(orderNo, itemNo, orderItemQuantity, user, totalPrice)//创建一个spy,可以用来做行为验证MockOrderEntity spyOrder = Spy(constructorArgs: [id, orderNo, itemNo, orderItemQuantity, null, user, totalPrice])//指定返回spyorderConverter.toEntity(_ as CreateOrderCommand) >> spyOrderLockInventoryCommand lockInventoryCommand = new LockInventoryCommand(itemNo, orderItemQuantity)when://触发测试Long resultId = orderService.createOrder(command)then://行为验证, 创建订单的同时,执行锁定库存lockInventory会被执行一次,同时会验证参数是否和我们提供lockInventoryCommand是否equals1 * inventoryService.lockInventory(lockInventoryCommand)//行为验证,最终订单执行insert1 * spyOrder.insert()//结果验证,验证返回的idresultId == id//状态验证spyOrder.orderStatus == OrderStatus.CREATE//以表格的形式提供测试数据集合where:orderNo | itemNo | orderItemQuantity | user | totalPrice"1" | "it" | 10 | "userA" | 9.9}}
5. 如何自动化执行单元测试
使用spock框架进行单测,可以通过添加maven插件,来在maven打包的时候自动执行单元测试代码。
<dependency><groupId>org.spockframework</groupId><artifactId>spock-core</artifactId><version>2.1-groovy-3.0</version><scope>test</scope></dependency><dependency><groupId>net.bytebuddy</groupId><artifactId>byte-buddy</artifactId><version>1.10.19</version><scope>test</scope></dependency><!-- Mandatory plugins for using Spock --><plugin><groupId>org.codehaus.gmavenplus</groupId><artifactId>gmavenplus-plugin</artifactId><version>1.12.0</version><executions><execution><goals><goal>compile</goal><goal>compileTests</goal></goals></execution></executions></plugin><plugin><artifactId>maven-surefire-plugin</artifactId><version>3.0.0-M5</version><configuration><includes><!-- 指定后缀为Test的文件,需要被执行单元测试 --><include>**/*Test</include></includes></configuration></plugin>
6. Spock测试框架中Mock,Stub,Spy的区别
Stub(桩对象):Stub对象用于模拟被测试对象的某些行为。Stub对象通常用来模拟一些外部依赖(interface)返回指定数据,以便于进行单元测试。不能用于用来做行为验证。
def ""() {given:def inventoryMapper = Stub(InventoryMapper)InventoryServce inventoryService = new InventoryServiceImpl(inventoryMapper)Inventory inv = new Inventory(10)inventoryMapper.selectById(_) >> invwhen://inventoryService.stockOut(quantity, id)inventoryService.stockOut(5, 1)then:inv.quantity == 5}
Mock(模拟对象):Mock对象和Stub对象类似,但是可以用来做行为验证,所以在spock中通常可以用mock替代stub。
def ""() {given:def inventoryMapper = Mock(InventoryMapper)InventoryServce inventoryService = new InventoryServiceImpl(inventoryMapper)Inventory inv = new Inventory(10)inventoryMapper.selectById(_) >> invwhen://inventoryService.stockOut(quantity, id)inventoryService.stockOut(5, 1)then://行为验证,inventoryMapper执行了一次stockOut1 * inventoryMapper.stockOut(_)inv.quantity == 5}
3. Spy(监视对象):上面的Stub,Mock都是创建一个假的实例,而Spy是在真实实例的基础上,类似创建一个包装类,它可以记录被测试对象的行为。既保留了原有实例功能的同时,还可以做行为验证。
```groovydef ""() {given:def inventoryMapper = Stub(InventoryMapper)InventoryServce inventoryService = new InventoryServiceImpl(inventoryMapper)Inventory inv = Spy(Inventory)inv.setQuantity(10)inventoryMapper.selectById(_) >> invwhen://inventoryService.stockOut(quantity, id)inventoryService.stockOut(5, 1)then://行为验证,inv执行了一次stockOut1 * inv.stockOut(_)inv.quantity == 5}
通常来说调用Spy对象的方法,会被默认委托给真实的对象来执行,即执行真实的方法,但是Spy同样也适用Stub行为,如:
def subscriber = Spy(SubscriberImpl, constructorArgs: ["Fred"])//Spy对象也可以像 Stub对象一样,替换掉receive方法,返回指定的值subscriber.receive(_) >> "ok"
7. Partial Mocks(部分Mock)
7.1 callRealMethod
通常来说Mock可以对class或者interface创建一个fake对象,不会执行真实的方法,当在写单元测试时有时会需要执行Mock对象的某些真实方法的时候,可以callRealMethod的方式来执行。
given:def subscriber = Mock(SubscriberImpl)//mock call方法subscriber.call(_) >> {return "called"}//通过callRealMethod指定mock对象执行原来的真实方法subscriber.receive(_) >> { callRealMethod() }then:subscriber.receive("")
7.2 spy
通过callRealMethod是一种方式,另一种,就是通过Spy来实现,因为Spy是基于真实的对象创建的,那么就可以反过来实现一个对象既可以调用真实方法,又可以调用假的方法。
given:def subscriber = Spy(SubscriberImpl, constructorArgs: ["Fred"])//mock call方法subscriber.call(_) >> {return "called"}then://这里会直接执行真实方法subscriber.receive("")
感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:
这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!有需要的小伙伴可以点击下方小卡片领取
相关文章:

单元测试的思考与实践
1. 什么是单元测试 通常来说单元测试,是一种自动化测试,同时包含一下特性: 验证很小的一段代码(业务意义 或者 代码逻辑 上不可再分割的单元),能够更准确的定位到问题代码的位置 能够快速运行(…...
C# Socket通讯简单Demo
C# Socket通讯简单Demo Client端Listener端 Client端 static void Main(string[] args) {XSocketService XSocketService new XSocketService();XSocketService.Init();while (true){Console.Write("请输入消息:");var msg Console.ReadLine();XSocket…...

视频融合共享平台LntonCVS视频监控管理平台技术方案详细介绍
LntonCVS国标视频综合管理平台是一款以视频为核心的智慧物联应用平台。它基于分布式、负载均衡等流媒体技术进行开发,提供广泛兼容、安全可靠、开放共享的视频综合服务。该平台具备多种功能,包括视频直播、录像、回放、检索、云存储、告警上报、语音对讲…...

C#ListView的单元格支持添加基本及自定义任意控件
功能说明 使用ListView时,希望可以在单元格显示图片或其他控件,发现原生的ListView不支持,于是通过拓展,实现ListView可以显示任意控件的功能,效果如下: 实现方法 本来想着在单元格里面实现控件的自绘的…...

数据库选型实践:如何避开分库分表痛点 | OceanBase用户实践
随着企业业务的不断发展,数据量往往呈现出快速的增长趋势。使用MySQL的用户面对这种增长,普遍选择采用分库分表技术作为应对方案。然而,这一方案常在后期会遇到很多痛点。 分库分表的痛点 痛点 1:难以保证数据一致性。由于分库分…...

3个火火火的AI项目,开源了!
友友们,今天我要给你们安利三个超酷的开源项目,它们都和AI有关,而且每一个都能让你的日常生活变得更加有趣和便捷!(最近AI绘图又又超神了,分享以下美图养眼) 01 字节出品,文字转语音Seed-TTS 字节推出了一…...
算法 | 子集数排列树满m叉树二分搜索归并排序快速排序
子集树:O(2^n) 一个序列的所有子集为2^n,即可看成具有2^n个叶节点的满二叉树 int backtrack(int k) //k表示扩展结点在解空间树中所处的层次 {if(k>n) //n标识问题的规模output(x); //x是存放当前解的一维数组if(constraint(k)…...

SpringBoot配置第三方专业缓存技术jetcache方法缓存方案
jetcache方法缓存 我们可以给每个方法配置缓存方案 JetCache 是一个基于 Java 的缓存库,支持多种缓存方案和缓存策略,主要用于提升应用程序的性能和响应速度。它提供了多种缓存模式和特性,可以根据需求选择合适的缓存方案。 JetCache 的主…...

游戏开发丨基于PyGame的消消乐小游戏
文章目录 写在前面PyGame消消乐注意事项系列文章写在后面 写在前面 本期内容:基于pygame实现喜羊羊与灰太狼版消消乐小游戏 下载地址:https://download.csdn.net/download/m0_68111267/88700193 实验环境 python3.11及以上pycharmpygame 安装pygame…...

软件项目管理概述
1.什么是项目? 2.项目管理的定义 3.项目管理的本质 4.项目成功的标志 5.项目管理的基本方法 6.项目的生命周期(启动 计划 执行 控制 结束) 7.结合生活中的某件事,谈谈项目管理的作用 项目管理在日常生活中扮演着重要的角色&…...

FastAdmin后台开发框架 lang 任意文件读取漏洞复现
0x01 产品简介 FastAdmin是一款基于PHPBootstrap的开源后台框架,专为开发者精心打造。它基于ThinkPHP和Bootstrap两大主流技术构建,拥有完善的权限管理系统和一键生成CRUD等强大功能。FastAdmin致力于提高开发效率,降低开发成本,…...
数字时代PLM系统的重要性
什么是 PLM(产品生命周期管理)? 从最基本的层面上讲,产品生命周期管理 (PLM)是管理产品从最初构思、开发、服务和处置的整个过程的战略流程。换句话说,PLM 意味着管理产品从诞生到消亡所涉及的一切。 什么是 PLM 软件…...

安卓实现圆形按钮轮廓以及解决无法更改按钮颜色的问题
1.实现按钮轮廓 在drawable文件新建xml文件 <shape xmlns:android"http://schemas.android.com/apk/res/android"<!--实现圆形-->android:shape"oval"><!--指定内部的填充色--><solid android:color"#FFFFFF"/><!-…...
常用原语介绍
1.在Xilinx的example(wavegen example)中看到他们的顶层模块的输入输出管脚都手动例化原语IBUF以及OBUF——工具也会自动给我们加上不必要自己加 2.非mrcc个srcc的管脚输入的时钟信号,无法进入mmcm和bufg————试验过会报错 3.实际上&…...

29. 透镜阵列
导论: 物理传播光学(POP)不仅可以用于简单系统,也可以设计优化复杂的光学系统,比如透镜阵列。 设计流程: 透镜阵列建模 在孔径类型中选择“入瞳直径”,并输入2 在视场设定中。设置一个视场&…...

深入理解并打败C语言难关之一————指针(3)
前言: 昨天把指针最为基础的内容讲完了,并且详细说明了传值调用和传址调用的区别(这次我也是做到了每日一更,感觉有好多想写的但是没有写完),下面不多废话,下面进入本文想要说的内容 目录&#…...

Ubuntu-24.04-live-server-amd64启用ssh
系列文章目录 Ubuntu-24.04-live-server-amd64安装界面中文版 Ubuntu安装qemu-guest-agent Ubuntu乌班图安装VIM文本编辑器工具 文章目录 系列文章目录前言一、输入安装命令二、使用私钥登录(可选)1.创建私钥2.生成三个文件说明3.将公钥复制到服务器 三…...
Leetcode 2786. 访问数组中的位置使分数最大(DP 优化)
Leetcode 2786. 访问数组中的位置使分数最大 DP 以每个位置为结尾的序列的分数取决于前方的分数,根据奇偶性计算,取最大值 超时 class Solution {public long maxScore(int[] nums, int x) {int n nums.length;long dp[] new long[n];Arrays.fill(dp…...

【docker实战】使用Dockerfile的COPY拷贝资源遇到的问题
事情是这样的。 在我负责的golang项目中,使用硬代码验证某块逻辑。比如: 于是,为了解决硬代码的问题,我制作了表格工具:【开源项目】Excel数据表自动生成工具v1.0版 – 经云的清净小站 (skycreator.top)。 使用表格工…...

如何用多线程执行 unittest 测试用例实现方案
前言 使用python做过自动化测试的小伙伴,想必都知道unittest和pytest这两个单元测试框架,其中unittest是python的官方库,功能相对于pytest来要逊色不少,但是uniitest使用上手简单,也受到的很多的小伙伴喜爱。一直以来都…...
rknn优化教程(二)
文章目录 1. 前述2. 三方库的封装2.1 xrepo中的库2.2 xrepo之外的库2.2.1 opencv2.2.2 rknnrt2.2.3 spdlog 3. rknn_engine库 1. 前述 OK,开始写第二篇的内容了。这篇博客主要能写一下: 如何给一些三方库按照xmake方式进行封装,供调用如何按…...
macOS多出来了:Google云端硬盘、YouTube、表格、幻灯片、Gmail、Google文档等应用
文章目录 问题现象问题原因解决办法 问题现象 macOS启动台(Launchpad)多出来了:Google云端硬盘、YouTube、表格、幻灯片、Gmail、Google文档等应用。 问题原因 很明显,都是Google家的办公全家桶。这些应用并不是通过独立安装的…...

Cinnamon修改面板小工具图标
Cinnamon开始菜单-CSDN博客 设置模块都是做好的,比GNOME简单得多! 在 applet.js 里增加 const Settings imports.ui.settings;this.settings new Settings.AppletSettings(this, HTYMenusonichy, instance_id); this.settings.bind(menu-icon, menu…...
Nginx server_name 配置说明
Nginx 是一个高性能的反向代理和负载均衡服务器,其核心配置之一是 server 块中的 server_name 指令。server_name 决定了 Nginx 如何根据客户端请求的 Host 头匹配对应的虚拟主机(Virtual Host)。 1. 简介 Nginx 使用 server_name 指令来确定…...
C++中string流知识详解和示例
一、概览与类体系 C 提供三种基于内存字符串的流,定义在 <sstream> 中: std::istringstream:输入流,从已有字符串中读取并解析。std::ostringstream:输出流,向内部缓冲区写入内容,最终取…...
鱼香ros docker配置镜像报错:https://registry-1.docker.io/v2/
使用鱼香ros一件安装docker时的https://registry-1.docker.io/v2/问题 一键安装指令 wget http://fishros.com/install -O fishros && . fishros出现问题:docker pull 失败 网络不同,需要使用镜像源 按照如下步骤操作 sudo vi /etc/docker/dae…...

RNN避坑指南:从数学推导到LSTM/GRU工业级部署实战流程
本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在聚客AI学院。 本文全面剖析RNN核心原理,深入讲解梯度消失/爆炸问题,并通过LSTM/GRU结构实现解决方案,提供时间序列预测和文本生成…...
大数据学习(132)-HIve数据分析
🍋🍋大数据学习🍋🍋 🔥系列专栏: 👑哲学语录: 用力所能及,改变世界。 💖如果觉得博主的文章还不错的话,请点赞👍收藏⭐️留言Ǵ…...
Android第十三次面试总结(四大 组件基础)
Activity生命周期和四大启动模式详解 一、Activity 生命周期 Activity 的生命周期由一系列回调方法组成,用于管理其创建、可见性、焦点和销毁过程。以下是核心方法及其调用时机: onCreate() 调用时机:Activity 首次创建时调用。…...

面向无人机海岸带生态系统监测的语义分割基准数据集
描述:海岸带生态系统的监测是维护生态平衡和可持续发展的重要任务。语义分割技术在遥感影像中的应用为海岸带生态系统的精准监测提供了有效手段。然而,目前该领域仍面临一个挑战,即缺乏公开的专门面向海岸带生态系统的语义分割基准数据集。受…...