当前位置: 首页 > news >正文

mit6.031软件构造 笔记 Testing

当你编码时,目标是使程序正常工作。 但作为测试设计者,你希望让它失败。 这是一个微妙但重要的区别。

为什么软件测试很难?

  • 做不到十分详尽:测试一个 32 位浮点乘法运算 。有 2^64 个测试用例!
  • 随机或统计测试效果差:其他工程学科可以测试小的随机样本(例如制造的硬盘驱动器的 1%)并推断整个生产批次的缺陷率。但是对于软件来说并非如此。软件行为在可能输入的空间内不连续且离散地变化,系统可能在广泛的输入范围内似乎工作正常,然后在单个边界点突然失效,堆栈溢出、内存不足错误和数字溢出错误往往会突然发生。

Test-first programming

顺序:

  1. Specification (规范):编写函数签名和注释,明确其行为、输入约束和输出。
  2. Test (测试):根据规范编写测试用例。
  3. Implementation (实现):编写实现代码,并通过已写的测试。

测试优先编程的最大好处是防止错误。 不要将测试留到开发结束,因为此时有一大堆未经验证的代码。 将测试留到最后只会使调试时间更长、更痛苦,因为错误可能存在于代码中的任何位置。

Systematic testing

我们希望进行系统测试,而不是详尽、随意或随机的测试。系统测试意味着我们以有原则的方式选择测试用例,目标是设计一个具有三个理想属性的测试套件:

通过分区选择测试用例

我们希望选择一组足够小的测试用例,以便于编写和维护并快速运行,但又足够彻底以发现程序中的错误。

为此,我们将程序的输入空间划分为多个子域,每个子域由一组输入组成。我们只需要为每个集合测试一个代表。 这种方法通过选择不同的测试用例,并强制测试探索随意或随机测试可能无法到达的输入空间区域,从而充分利用了有限的测试资源。

例:

Math.abs()

测试用例:

  • a = 17 覆盖子域 a > 0
  • a = 0 覆盖子域 a = 0
  • a = -3 覆盖子域 a < 0

Math.max()

测试用例:

  • (a,b) = (1, 2) 覆盖 a < b
  • (a,b) = (10, -8) 覆盖 a > b
  • (a,b) = (9, 9) 覆盖 a = b

子域应具有三个理想的属性:

  • 互斥
  • 完整
  • 非空

自动化单元测试

  • 单元测试 (Unit Test):测试单个模块(如函数)的测试。
  • 自动化:使用测试框架(如Mocha for JS/TS)编写测试代码,自动运行并检查结果(使用assert.strictEqual, assert.deepStrictEqual等断言),输出通过/失败报告。
  • 文档化测试策略 (Documenting Strategy):在测试代码中以注释形式记录所采用的分区策略,并为每个测试用例命名其所覆盖的子域(如it("covers a < b", ...)),这极大地增强了测试套件的可理解性。

黑盒 vs 玻璃盒测试

  • 黑盒测试仅根据规范选择测试用例,不查看实现代码。这是测试优先编程的天然方式。
  • 玻璃盒测试基于对实现代码的了解选择测试用例(例如,测试不同的算法分支、内部缓存机制等)。
  • 结合使用:先进行黑盒测试(定义分区),再通过玻璃盒测试和覆盖率分析来补充测试用例,提高彻底性。

覆盖率

衡量测试套件对代码的覆盖程度,常用指标:

  • Statement coverage (语句覆盖):是否每条语句都被至少一个测试执行过?(常见目标)
  • Branch coverage (分支覆盖):是否每个控制分支(如if/else的两边)都被至少一个测试执行过?
  • Path coverage (路径覆盖):是否所有可能的执行路径都被覆盖?使用工具(如Istanbul/nyc)测量覆盖率,并补充测试用例以提高覆盖率

单元测试 vs. 集成测试

  • 单元测试孤立地测试单个模块。优点:错误更容易定位(就在被测试的模块中)。
  • 集成测试:测试多个模块的组合或整个系统。必要但错误可能出现在任何连接的模块中。
  • 策略:首先依靠全面的单元测试建立对各个模块的信心,然后使用集成测试来检查模块间的交互。尽量避免在单元测试中依赖其他可能出错的模块

自动化回归测试

  • 回归测试 (Regression Testing):在每次修改代码后(修复bug、添加功能、优化性能)运行完整的测试套件,防止修改引入新的错误
  • 测试优先调试 发现bug时,立即编写一个能重现该bug的测试用例,并将其加入测试套件。修复bug后,该测试用例就成为防止未来回归的回归测试
  • 自动化是回归测试可行的关键。

迭代式测试优先编程

软件开发不是线性的,应采用迭代方式:

  1. 编写初步规范和测试。
  2. 编写初步实现。
  3. 根据实现中发现的问题,迭代改进规范、测试和实现。
    迭代允许更快地获得反馈,更有效地利用时间,特别是在解决复杂问题时。
http://www.dtcms.com/a/360228.html

相关文章:

  • ROI、 binning、下采样功能区别
  • windows编译minicap.so文件
  • 由题构造 嵌入汇编(汇编)
  • NAS Docker 安装N8N
  • 计算机视觉与深度学习 | 双目立体特征提取与匹配算法综述——理论基础、OpenCV实践与MATLAB实现指南
  • 猛犸Lark max 无线麦克风录音爆音问题的解决
  • 【STC库函数】使用芯片自带的EEPROM来保存掉电不丢失的数据
  • 开发常用工具专栏
  • 赵玉平《梁山政治》读书笔记(下部)
  • wifi控制舵机
  • WinExec
  • Nginx反向代理及配置
  • c++ 线程局部存储(Thread-Local Storage,TLS)
  • Langflow Memory 技术深度分析
  • java--浅拷贝深拷贝
  • Introduction to GIS —— Chapter 3(Vector Data Model)
  • 雪花算法生成分布式ID
  • AI 智能体汇总,自动执行任务的“真 Agent”
  • 动态规划入门(三):一些经典动态规划模型
  • 赵玉平《刘备谋略》读书笔记(下部)
  • 小迪自用web笔记22
  • 01背包day35
  • 设计模式 | 常见的设计模式(单例、工厂、代理、适配器、责任链等等)
  • VisionProC#联合编程火花塞距离检测与VisionPro操作
  • libmodbus库,c++配置方法
  • 【CUDA入门·Lesson 1】Ubuntu实战:CUDA 概念、nvidia-smi 工具与 GPU 参数详解
  • 在Unity中,让子物体不随父物体移动或转动的方法!
  • 下一代防火墙(NGFW):从定义到功能
  • 试试 Xget 加速 GitHub 克隆仓库
  • 【WEB】[BUUCTF] <GXYCTF2019禁止套娃>《php函数的运用》