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

程序员思维体操:TDD修炼手册

程序员思维体操:TDD修炼手册

——从"先写代码"到"测试先行"的认知革命

一、重新认识TDD:不仅仅是写测试

什么是TDD(测试驱动开发)
TDD其实很简单,不要看名字很高级复杂,传统开发是直接开发功能,TDD则是先写好测试再开发功能。具体来说:

  1. 开发前先编写描述功能的测试用例
  2. 编写刚好让测试通过的代码
  3. 重构代码使其更优雅,同时保持测试通过

某电商系统开发时,团队在实现优惠券功能前,先写下了这样的测试:

@Test  
public void20050时支付金额正确() {  Coupon coupon = new Coupon("FULL_200_OFF_50");  Order order = new Order(250.00);  order.applyCoupon(coupon);  assertEquals(200.00, order.getPayAmount());  
}  

这个测试用例就像施工图纸,明确限定了代码的行为边界。

TDD的设计哲学

  • 需求即测试:将模糊的需求转化为可验证的断言
  • 小步快跑:每次只实现一个微小功能点(如先处理整数相加,再考虑浮点数)
  • 安全网思维:测试集是代码的防弹衣,重构时不再如履薄冰
  • 设计驱动:测试倒逼模块解耦,天然符合SOLID原则

二、TDD实战四部曲:手把手教你开车

1. 红绿灯循环:程序员的新节奏

Step1:红灯阶段(编写失败测试)
在IDE新建文件StringCalculatorTest.java,写下:

@Test  
public void 空字符串返回0() {  assertEquals(0, StringCalculator.add(""));  
}  

此时运行测试必然报错——因为StringCalculator类还不存在。

Step2:绿灯冲刺(最小实现)
创建StringCalculator.java,仅实现能让测试通过的最简代码:

public class StringCalculator {  public static int add(String numbers) {  return 0;  }  
}  

虽然这明显是个"作弊"实现,但此刻测试已变绿。

Step3:重构进化(优化设计)
新增测试输入"1"应返回1,迫使代码升级:

public static int add(String numbers) {  return numbers.isEmpty() ? 0 : Integer.parseInt(numbers);  
}  

通过不断添加测试驱动功能迭代,最终实现完整计算器。

2. 测试用例设计心法

案例:开发简易购物车

  • 基础路径
    @Test  
    public void 添加3件单价100商品总价300() {  Cart cart = new Cart();  cart.addItem(new Item("水杯", 100), 3);  assertEquals(300, cart.getTotalPrice());  
    }  
    
  • 边界条件
    @Test  
    public void 添加0件商品时应抛出异常() {  assertThrows(InvalidQuantityException.class,  () -> cart.addItem(item, 0));  
    }  
    
  • 异常场景
    @Test  
    public void 库存不足时无法添加商品() {  Item limitedItem = new Item("限定款", 999, 1);  // 最后1件库存  cart.addItem(limitedItem, 1);  assertThrows(InventoryShortageException.class,  () -> cart.addItem(limitedItem, 1));  
    }  
    
3. 破解复杂依赖:Mock技术实战

开发支付模块时,如何在不调用真实银行接口的情况下测试?

@Test  
public void 支付失败时应记录日志() {  // 创建模拟支付网关  PaymentGateway mockGateway = mock(PaymentGateway.class);  when(mockGateway.pay(any())).thenReturn(false);  PaymentService service = new PaymentService(mockGateway);  service.processPayment(new Order(100.00));  // 验证是否调用日志记录  verify(logger).error("支付失败,订单号:123");  
}  

通过Mockito等框架,可以隔离外部系统专注业务逻辑验证。

三、日常训练计划:从菜鸟到TDD武者

1. 新手村任务(第1周)
  • LeetCode特训
    选择简单题目(如两数之和),强制使用TDD流程:

    1. 先写测试用例
    2. 实现最笨解法
    3. 重构优化时间复杂度
  • 代码考古
    在GitHub找TDD风格的开源项目(如JUnit自身),观察测试与代码比例

2. 高手试炼(持续进阶)
  • 改造遗留系统
    选择公司旧模块,为其补充测试覆盖率(从30%提升到70%)
  • 极限挑战
    尝试用TDD开发贪吃蛇游戏,测试用例包括:
    @Test  
    public void 蛇头碰到墙时游戏结束() {  snake.move(Direction.UP);  assertTrue(game.isOver());  
    }  
    
  • 模式融合
    在TDD过程中自然引入设计模式,例如:
    // 测试观察者模式  
    @Test  
    public void 商品降价时通知所有用户() {  Product iphone = new Product("iPhone15", 7999);  User zhangsan = new User("张三");  iphone.addObserver(zhangsan);  iphone.setPrice(6999);  assertTrue(zhangsan.getNotifications().contains("iPhone15降价啦!"));  
    }  
    

四、避坑指南:TDD实践中的暗礁

1. 测试过度症

错误案例
为Getter/Setter方法编写测试
解决方案
遵循"测试行为而非实现"原则,只验证业务逻辑

2. 速度焦虑症

典型症状
认为写测试拖慢进度,退回老路
数据支撑
谷歌统计显示,TDD初期效率降低20%,但后期维护时间减少50%

3. 用例脆弱症

反面教材

@Test  
public void 测试列表顺序() {  List<String> list = getData();  assertEquals("苹果", list.get(0));  // 一旦排序逻辑改变就失败  
}  

改进方案
断言集合包含元素而非固定顺序:

assertTrue(list.containsAll(Arrays.asList("苹果", "香蕉")));  

结语:测试先行的思维革命

当你在设计数据库表结构前先写下UserRegistrationTest,当看到产品文档时脑中自动浮现测试用例树,当代码评审会上能指着测试集说"这就是需求文档"——这一刻,你已完成了从代码工人到软件工匠的蜕变。

TDD是迄今为止最强大的代码质量提升工具,但需要勇气直面初期的不适。

相关文章:

程序员思维体操:TDD修炼手册

程序员思维体操&#xff1a;TDD修炼手册 ——从"先写代码"到"测试先行"的认知革命 一、重新认识TDD&#xff1a;不仅仅是写测试 什么是TDD&#xff08;测试驱动开发&#xff09; TDD其实很简单&#xff0c;不要看名字很高级复杂&#xff0c;传统开发是直…...

PHP异常处理__RuntimeException运行时错误

以下是对 PHP 中 RuntimeException 的详细解释&#xff1a; 一、RuntimeException 概述 RuntimeException 是 PHP 内置的异常类&#xff0c;它继承自 Exception 类。它通常用于表示在程序运行时发生的异常情况&#xff0c;这些异常情况通常是在程序正常执行过程中出现的错误&…...

从性能到安全:大型网站系统架构演化的 13 个核心维度

大型网站系统架构的演化是一个复杂的过程&#xff0c;涉及到多个维度的技术内容&#xff0c;从关键维度进行详细分析&#xff1a; 1.性能维度 缓存技术&#xff1a;包括浏览器缓存、CDN&#xff08;内容分发网络&#xff09;缓存、服务器端缓存&#xff08;如 Memcached、Red…...

基于PaddleOCR对图片中的excel进行识别并转换成word优化(二)

0、原图 一、优化地方 计算行的时候&#xff0c;采用概率分布去统计差值概率比较大的即为所要的值。 def find_common_difference(array):"""判断数组中每个元素的差值是否相等&#xff0c;并返回该差值:param array: 二维数组&#xff0c;其中每个元素是一个…...

spring Ai---向量知识库(二)

RAG&#xff1a;检索增强&#xff0c;结合了检索和生成两种技术&#xff1b;用于提升生成模型的效果。 1.信息检索&#xff08;R) &#xff1a;系统从一个大型文档库中检索出与查询最相关的文档片段。这一步的目标是找到那些可能包含答案或相关信息的文档。 2.生成增强&#xf…...

Nvidia显卡架构演进

1 简介 显示卡&#xff08;英语&#xff1a;Display Card&#xff09;简称显卡&#xff0c;也称图形卡&#xff08;Graphics Card&#xff09;&#xff0c;是个人电脑上以图形处理器&#xff08;GPU&#xff09;为核心的扩展卡&#xff0c;用途是提供中央处理器以外的微处理器帮…...

rollup使用讲解

rollup 总结 什么是 rollup? rollup 是一个 JavaScript 模块打包器,在功能上要完成的事和 webpack 性质一样,就是将小块代码编译成大块复杂的代码,例如 library 或应用程序。在平时开发应用程序时,我们基本上选择用 webpack,相比之下,rollup.js 更多是用于 library 打…...

USO服务器操作系统手动升级GCC 12.2.0版本

1. 从 GNU 官方 FTP 服务器下载 GCC 12.2.0 的源码包&#xff0c;并解压进入源码目录。 wget https://ftp.gnu.org/gnu/gcc/gcc-12.2.0/gcc-12.2.0.tar.gz tar -zxvf gcc-12.2.0.tar.gz cd gcc-12.2.0 2. 运行脚本下载并配置 GCC 编译所需的依赖库。此步骤会自动下载如 GMP…...

STM32F407使用ESP8266实现阿里云OTA(上)

文章目录 前言一、阿里云OTA二、命令调试1.升级包上传2.OTA订阅和上报的主题3.命令调试4.具体效果三、所用到的工具和材料前言 在经过前面对ESP8266、SD卡、FLASH的了解之后,终于要进入我们的正题了,就是使用STM32和ESP8266实现阿里云的OTA。这一功能并不复杂,只是需要主要…...

玩转Docker | 使用Docker部署DashMachine个人书签工具

玩转Docker | 使用Docker部署DashMachine个人书签工具 前言一、DashMachine介绍DashMachine简介DashMachine使用场景二、系统要求环境要求环境检查Docker版本检查检查操作系统版本三、部署DashMachine服务下载镜像创建容器创建容器检查容器状态检查服务端口安全设置四、访问Das…...

测试基础笔记第九天

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 一、数据类型和约束1.数据类型2.约束3.主键4.不为空5.唯一6.默认值 二、数据库操作1.创建数据库2.使用数据库3.修改数据库4.删除数据库和查看所有数据库5.重点&…...

使用n8n构建自动化工作流:从数据库查询到邮件通知的使用指南

n8n是一款强大的开源工作流自动化工具&#xff0c;可以帮助你将各种服务和应用程序连接起来&#xff0c;创建复杂的自动化流程。下面我将详细介绍一个实用的n8n用例&#xff1a;从MySQL数据库查询数据并发送邮件通知&#xff0c;包括使用场景、搭建步骤和节点部署方法。 使用场…...

Python爬虫与代理IP:高效抓取数据的实战指南

目录 一、基础概念解析 1.1 爬虫的工作原理 1.2 代理IP的作用 二、环境搭建与工具选择 2.1 Python库准备 2.2 代理IP选择技巧 三、实战步骤分解 3.1 基础版&#xff1a;单线程免费代理 3.2 进阶版&#xff1a;多线程付费代理池 3.3 终极版&#xff1a;Scrapy框架自动…...

Unity 将Excel表格中的数据导入到Mysql数据表中

1.Mysql数据表users如下&#xff1a; 2.即将导入的Excel表格如下&#xff1a; 3.代码如下&#xff1a; using System; using System.Data; using System.IO; using Excel; using MySql.Data.MySqlClient; using UnityEngine; using UnityEditor;public class ImportExcel {// …...

JavsScript 原型链

解决构造函数浪费内存的问题 每一个构造函数都有一个属性prototype属性&#xff0c;指向一个原型对象 原型是构造函数的一个属性 prototype 给数组类型扩展 正常代码&#xff1a; prototype中的this指向为调用对象 所以 基本关系&#xff1a;构造函数产生两个部分&…...

MySQL 索引:深度解析与高效使用

MySQL 索引:深度解析与高效使用 引言 MySQL 是一种广泛使用的开源关系型数据库管理系统,其强大的功能和性能使其成为众多应用程序的首选数据库。在 MySQL 中,索引是提高查询效率的关键因素之一。本文将深入探讨 MySQL 索引的概念、类型、创建、优化以及注意事项,帮助您更…...

消息中间件RabbitMQ02:账号的注册、点对点推送信息

一、默认用户登录和账号注册 1.登录 安装好了RMQ之后&#xff0c;我们可以访问如下地址&#xff1a; RabbitMQ Management 输入默认的管理员密码&#xff0c;4.1.0的管理员账号和密码是&#xff1a; guest guest 2.添加账号 consumer consumer 添加成功后&#xff1a; 角色…...

大语言模型的评估指标

目录 一、混淆矩阵 1. 混淆矩阵的结构&#xff08;二分类为例&#xff09; 2.从混淆矩阵衍生的核心指标 3.多分类任务的扩展 4. 混淆矩阵的实战应用 二、分类任务核心指标 1. Accuracy&#xff08;准确率&#xff09; 2. Precision&#xff08;精确率&#xff09; 3. …...

Python 设计模式:模板模式

1. 什么是模板模式&#xff1f; 模板模式是一种行为设计模式&#xff0c;它定义了一个操作的算法的骨架&#xff0c;而将一些步骤延迟到子类中。模板模式允许子类在不改变算法结构的情况下&#xff0c;重新定义算法的某些特定步骤。 模板模式的核心思想是将算法的固定部分提取…...

HSTL详解

一、HSTL的基本定义 HSTL&#xff08;High-Speed Transceiver Logic&#xff09; 是一种针对高速数字电路设计的差分信号接口标准&#xff0c;主要用于高带宽、低功耗场景&#xff08;如FPGA、ASIC、高速存储器接口&#xff09;。其核心特性包括&#xff1a; 差分信号传输&…...

好用————python 库 下载 ,整合在一个小程序 UIUIUI

上图~ import os import time import threading import requests import subprocess import importlib import tkinter as tk from tkinter import ttk, messagebox, scrolledtext from concurrent.futures import ThreadPoolExecutor, as_completed from urllib.parse im…...

OpenVINO教程(五):实现YOLOv11+OpenVINO实时视频目标检测

目录 实现讲解效果展示完整代码 本文作为上篇博客的延续&#xff0c;在之前实现了图片推理的基础上&#xff0c;进一步介绍如何进行视频推理。 实现讲解 首先&#xff0c;我们需要对之前的 predict_and_show_image 函数进行拆分&#xff0c;将图像显示与推理器&#xff08;pre…...

CentOS的安装以及网络配置

CentOS的下载 在学习docker之前&#xff0c;我们需要知道的就是docker是运行在Linux内核之上的&#xff0c;所以我们需要Linux环境的操作系统&#xff0c;当然了你也可以选择安装ubuntu等操作系统&#xff0c;如果你不想在本机安装的话还可以考虑买阿里或者华为的云服务器&…...

【初级】前端开发工程师面试100题(一)

本题库共计包含100题&#xff0c;考察html&#xff0c;css&#xff0c;js&#xff0c;以及react&#xff0c;vue&#xff0c;webpack等基础知识掌握情况。 HTML基础篇 说说你对HTML语义化的理解&#xff1f; 语义化就是用合适的标签表达合适的内容&#xff0c;比如<header&…...

eplan许可证与防火墙安全软件冲突

在使用EPLAN电气设计软件时&#xff0c;有时会遇到许可证与防火墙或安全软件之间的冲突。这种冲突可能导致许可证无法激活或软件无法正常运行&#xff0c;给用户带来诸多不便。本文将为您解析EPLAN许可证与防火墙/安全软件冲突的原因&#xff0c;并提供解决方案&#xff0c;帮助…...

「Java EE开发指南」用MyEclipse开发EJB 3无状态会话Bean(二)

本教程介绍在MyEclipse中开发EJB 3无状态会话bean&#xff0c;由于JPA实体和EJB 3实体非常相似&#xff0c;因此本教程不涉及EJB 3实体Bean的开发。在本教程中&#xff0c;您将学习如何&#xff1a; 创建EJB 3项目创建无状态会话bean部署并测试bean 在上文中&#xff08;点击…...

Stable Diffusion秋叶整合包V4独立版Python本地API连接指南

秋叶整合包V4独立版Python本地API连接指南 秋叶整合的Stable Diffusion V4独立版支持通过Python调用本地API实现自动化图像生成。以下是具体操作流程及注意事项&#xff1a; 一、启用API服务 启动器配置 • 在秋叶启动器的 高级选项 中添加以下参数&#xff1a; --api --liste…...

小程序 GET 接口两种传值方式

前言 一般 GET 接口只有两种URL 参数和路径参数 一&#xff1a;URL 参数&#xff08;推荐方式&#xff09; 你希望请求&#xff1a; https://serve.zimeinew.com/wx/products/info?id5124接口应该写成这样&#xff0c;用 req.query.id 取 ?id5124&#xff1a; app.get(&…...

深度学习在DOM解析中的应用:自动识别页面关键内容区块

摘要 本文介绍了如何在爬取东方财富吧&#xff08;https://www.eastmoney.com&#xff09;财经新闻时&#xff0c;利用深度学习模型对 DOM 树中的内容区块进行自动识别和过滤&#xff0c;并将新闻标题、时间、正文等关键信息分类存储。文章聚焦爬虫整体性能瓶颈&#xff0c;通…...

PyQt6实例_pyqtgraph多曲线显示工具_代码分享

目录 概述 效果 代码 返回结果对象 字符型横坐标 通用折线图工具 工具主界面 使用举例 概述 1 分析数据遇到需要一个股票多个指标对比或一个指标多个股票对比&#xff0c;涉及到同轴多条曲线的显示&#xff0c;所以开发了本工具。 2 多曲线显示部分可以当通用工具使…...