先测试再开发?TDD测试驱动开发了解一下?

article/2025/10/11 19:54:03

1、什么是TDD

我第一次接触TDD这个概念,是在<<代码整洁之道>>中,作者鲍勃大叔在书中,写了一些关于测试代码的代码规范,其实就提到了有关TDD三定律:

- 定律一: 在编写不能通过的单元测试前,不可编写生产代码
- 定律二: 只可编写刚好无法通过的单元测试,不能编译也算不能通过
- 定律三:只可编写刚好足以通过当前失败测试的生存代码

我第一次读到这三个定律时,不能说是毫无头绪,只能说是一脸懵逼。
在这里插入图片描述
完全不知道作者想表达啥意思,也没有案例代码。
对此,我不得不网上查阅的很多相关文章,最后总结出来。

TDD测试驱动开发,就是先写测试用例,再去开发功能。
这里测试驱动开发里的驱动是做动词,不是名词

好了,现在如果别人问你TDD是什么,你就可以直接这样告诉他。

2、传统开发方式

我们传统开发一个功能是这么开发的?
传统编码方式

  • 需求分析,想不清楚细节,管他呢,先开始写
  • 发现需求细节不明确,去跟业务人员确认
  • 确认好几次终于写完所有逻辑
  • 运行起来测试一下,靠,果然不工作,调试
  • 调试好久终于工作了
  • 转测试,QA 测出 bug,debug, 打补丁
  • 终于,代码可以工作了
  • 一看代码烂的像坨屎,不敢动,动了还得手工测试,还得让 QA 测试,还得加班…

传统的开发方式,都是以开发为主,直接开始编写代码,代码出了问题,再去改,多改几次,你就会觉得这代码简直就是屎山,想重构一下,又怕出新的问题。

3、TDD步骤

而TDD测试驱动开发是怎么做的呢?

TDD要求我们先根据需求去拆分任务,把一个大的任务拆分位各个模块,也就是一个个的函数,我们再去为这些函数去编写最小的测试,再去写能让这个最小的测试通过的最小的实现。
TDD的生命周期图如下。
在这里插入图片描述

这样做的好处是:

  • 1、有助于我们提前澄清需求
  • 2、可以通过单元测试断言的诊断机制快速得出反馈
  • 3、当我们写完了所有的需求,会发现所有的需求都会被测试覆盖了

4、举个例子

正所谓,光说不练,假把式;下来我们来整个简单的例子去理解一下测试驱动开发;
假如我需要写个功能,分析用户上传的文本中,每个单词的数量,并且按照数量倒序排序,这个应该怎么实现:
比如说文本如下:

Hello world
Hello CSDN
Hello Boy
My name is Boy 
is is 

那输出就是:

Hello 3
is 3
Boy 2
world 1
CSDN 1
My 1 
name 1

如果是新手或者是完全不懂代码设计的人拿到这样的功能,可能会这样写:

 public static void main(String[] args) throws Exception{String words = "";File file=new File("word.txt");Scanner sc=new Scanner(file);while(sc.hasNextLine()){String line=sc.nextLine();words = words + line + " ";}System.out.println(words);String[] wordArrays =  words.split(" ");HashMap<String,Integer> hashMap = new HashMap<>();for(int i=0;i<wordArrays.length;i++){Set<String> wordSet = hashMap.keySet();if(wordSet.contains(wordArrays[i])){Integer number=hashMap.get(wordArrays[i]);number++;hashMap.put(wordArrays[i],number);}else{hashMap.put(wordArrays[i],1);}}System.out.println("统计单词------------------");List<Map.Entry<String, Integer>> list = new ArrayList<Map.Entry<String, Integer>>(hashMap.entrySet()); //转换为listlist.sort(new Comparator<Map.Entry<String, Integer>>() {@Overridepublic int compare(Map.Entry<String, Integer> o1, Map.Entry<String, Integer> o2) {return o2.getValue().compareTo(o1.getValue());}});System.out.println(list);       }

好了,写完了,让我们运行一下:
在这里插入图片描述
运行结果貌似也没有问题,好了提交代码。
真的没有问题吗?统计单词的数目也没有问题,但是,如果后期,你要对这段代码做维护,要去修改这段代码,这段代码读起来是什么感觉?在我看来,这段代码,就是屎山。
代码里几乎完全没有注释,读这段代码,得从头开始往下读,如果其中一段代码出了问题,你必须在整段代码中寻找错误,非常浪费时间。

让我们来试试看TDD的写法,TDD要求首先要把一个功能,去拆分分成各个小功能,然后去为这些小功能写测试用例。
这个统计单词数目的代码应该怎么拆分,试着拆分成小功能一下(这里要注意一下,同样的功能,在拆分模块的时候,不同的人选择的拆分方法可能不同,一千个人里有一千个哈姆雷特,一千个人里也有一千种拆分方法)

从外部文件读取单词到字符串
分隔单词到数组
用haspmap分组
按照排序规则进行输出

拆分好后,我们就可以为这些功能编写测试用例了,
我们先编写测试用例,用assert断言测试用例是否通过,运行,我们可以看出,方法还没有进行开发,显示未通过。
在这里插入图片描述

这一步,就是TDD定律二中规定的

- 定律二: 只可编写刚好无法通过的单元测试,不能编译也算不能通过

下一步要做什么?我们看定律一

- 定律一: 在编写不能通过的单元测试前,不可编写生产代码

定律一反过来是什么意思,不就是编写好不能通过的单元测试后,就可以开始编写生产代码了吗?
于是开始编写生产代码进行测试
在这里插入图片描述

在这里,我们编写好了方法,在执行测试用例后,显示绿色,代表测试用例通过。
这时候又满足了第三定律

- 定律三:只可编写刚好足以通过当前失败测试的生存代码

我们的测试用例,专门测试一个小的功能,只为了通过这个方法。

下一步我们再重复以上步骤,去TDD其他模块。
直到TDD完所有模块,我们的功能就开发完了。
代码如下:

 public static void main(String[] args) throws Exception {//读取文件File file=new File("word.txt");String words = readFile(file);String[] s1 = words.split(" ");//单词记录到hashmapHashMap<String, Integer> stringIntegerHashMap = groupHashMap(s1);//排序List<Map.Entry<String, Integer>> entries = orderHashMap(stringIntegerHashMap);//输出System.out.println(entries);}

方法的代码太长,太展篇幅就不粘贴了,你看我们新的main方法,代码就比较简介,如果出了问题,只可能在这三个方法中其中,我们可以快速定位到方法中去,并且可以用之前编写的测试用例进行测试。

5、总结

关于TDD测试驱动开发,我感觉精髓就是对功能进行拆分,用测试用例去测试功能,我相信很多人,都不会去编写测试用例,代码写好后,就去页面上点几下,其实这是不太好的。因为如果这里的功能改动比较频繁,你每次去页面上通过点击的方式测试功能,你得打开浏览器,登录地址,寻找IP,为功能配置参数,这一套下来,真的非常耗费时间。
一次如果5分钟,10次就是50分钟。
而TDD建议的是什么?建议通过测试用例的方式去测试,它要求你必须编写好测试用例后再去写代码,这样就能保住,你每个小功能,都有一个测试用例,这样,之后你改一个地方,只要找到这个地方所对应的测试用例就能测试了,非常方便。
当然,TDD这种开发方法其实弊端也是很明显的,比如,大多数程序员,其实是怎么做测试的?就是直接重启项目,去页面上看看,功能对不对,测试用例?那是什么?我不是开发吗,我又不是测试。我去页面点几下测试,可能只需要几分钟,我去配置测试用例,八成等得二十分钟,所以大多数程序员可能还是会选择通过页面点击的方式去测试。

测试用例真的是没有必要的吗?如果你去新建一个Maven项目,你会发送,test目录和main目录是同一级别的
在这里插入图片描述
我相信在设计的时候,可能设计者(设计Maven的程序员肯定是大佬中的大佬),也是认为测试用例是非常重要的,才会把test目录放在这个位置吧。


http://chatgpt.dhexx.cn/article/MdkOsPG3.shtml

相关文章

测试驱动开发(TDD)前端篇

当你在写生产代码时&#xff0c;你处在高认知的状态&#xff08;obvious&#xff09;&#xff0c;你的研发流程和你的工程实践&#xff0c;有助于你一步一步的提升你的认知能力&#xff0c;把你的问题进行一个降解&#xff08;分解&#xff09;&#xff0c;只要你做到同样的事情…

TDD 开发测试

测试驱动开发(Test-Driven Development)。是敏捷开发中的一项核心实践和技术。 TDD是在开发功能代码之前&#xff0c;先编写单元测试用例代码&#xff0c;测试代码确定需要编写什么产品代码。 变红 ——> 变绿 ——> 重构 在进行 TDD 案例编写的时候&#xff0c;看一…

TDD测试驱动开发

TDD测试驱动开发 什么是测试驱动开发测试驱动开发该怎么做&#xff1f;需求一&#xff1a;输入一个非元音字符&#xff0c;并预期返回字符本身 (输入"h" 返回“h”)需求二&#xff1a;输入一个元音&#xff08;a,e,i,o,u&#xff09;&#xff0c;返回 mommy (输入&qu…

TDD测试驱动学习

gtest 和 gmock 安装 //如果不知道对应库名字可以执行这个命令查找对应库,如果没找到要去跟新对应的源sudo apt update sudo apt-cache search gtest sudo apt-cache search gmock 测试例子 #include <string> using std::string;// 定义 Soundex 类 class Soundex {pub…

TDD (test driver development)测试驱动开发

##为什么需要测试驱动/或者说需要单元测试 我们工作接触的软件项目&#xff0c;不是学生时代&#xff0c;玩一玩就不管了&#xff0c;工作的项目&#xff0c;需要长期维护&#xff0c;并且随着时间的推移需要增加新的需求&#xff0c;进行修改&#xff0c;优化。此时已经距离你…

tdd(测试驱动开发)的概述

最近的工作的项目&#xff0c;使用了tdd&#xff08;test-driven development测试驱动开发&#xff09;的开发模式。 这两几年大概听说了无数种xxx-dd, ddd, tdd, atdd, bdd, fdd, udd各种名词眼花缭乱&#xff0c;当然很多dd其实也有相互借鉴&#xff08;抄袭&#xff09;的部…

测试驱动开发(TDD)实践与技巧

文章目录 引言Google Mock测试用例结构断言经典式断言Hamcrest 断言 测试驱动开发&#xff1a;第一个示例开场白开始吧去掉不干净的代码增量性fixture 设置思索与测试驱动开发测试驱动与测试 测试驱动开发基础与单元测试单元测试的组织结构测试驱动开发周期&#xff1a;红-绿-重…

opencv配置相关的截图参考

opencv配置相关的截图参考&#xff0c;如下&#xff1a;

Anaconda3安装及opencv配置

一、Anaconda安装 1.直接百度搜anaconda&#xff0c;进入官网即可&#xff08;anaconda网站链接&#xff09; 2.点击图片上黑框&#xff08;Get Started&#xff09;即可进入下一步&#xff0c;选择下图中第四个 3.选择适合电脑类型的anaconda安装器&#xff0c;注意选好64位…

linux安装配置opencv

刚开始学习ubuntu&#xff0c;有些项目需要用到opencv&#xff0c;当我用下面的命令安装包的时候&#xff0c;总是出现“E&#xff1a;无法定位软件包 opencv”的错误。然后开始着手解决&#xff0c;网上搜集了很多的教程&#xff0c;大部分都是说要更换源&#xff0c;我也照做…

opencv安装配置测试

前面安装了pcl和qt&#xff0c;以及qt中的vtk&#xff0c;这里配置下opencv4.3 将cv复制到D盘下。 安装完成之后&#xff0c;添加环境变量&#xff0c;[计算机]->右键 [属性]->[高级系统设置]->[环境变量]->[系统环境变量]->编辑 [Path]&#xff0c;添加“D:…

Qt中配置OpenCV

Qt中配置OpenCV 1. 环境下载2. 进行编译和安装2.1 新建opencv-build文件夹&#xff0c;用于opencv的 编译和安装&#xff08;直接在opencv下新建即可&#xff09;2.2 CMake设置2.3 命令行进行编译安装 3. Qt测试4. 其它问题4.1 若电脑上安装有PyQt4.2 电脑无法访问wai网 1. 环境…

Anaconda配置OpenCV

文章目录 1.安装Anaconda2.配置OpenCV2.1打开Anaconda Prompt2.2找到里面Scripts的路径2.3下载2.4验证是否配置成功 3.出错解决办法4.参考文章 1.安装Anaconda 可以查看我的上一篇文章&#xff1a;Anaconda下载、安装和环境配置 2.配置OpenCV 2.1打开Anaconda Prompt 在开始…

Opencv学习笔记——opencv配置安装与IDE环境安装

文章目录 前言一、opencv配置安装二、IDE的安装总结 前言 既然开始对AI视觉这个方面有兴趣&#xff0c;也初步接触了一些AI视觉在嵌入式方面的应用&#xff0c;那自然少不了对Opencv的学习。到现在开始学习opencv之前对它了解不多&#xff0c;只知道opencv的全称是Open Source…

vs + python + opencv 配置

首要条件&#xff0c;在vs上安装python环境。 以Visual Studio 2019为例讲解如何配置python、opencv、及相关第三方库。&#xff08;其它vs版本只是在界面上有所区别&#xff0c;过程相同。&#xff09; 步骤一&#xff1a;安装python开发工具 按下图操作&#xff0c;勾选Pyt…

vscode配置opencv

前言 本篇文章主要用来记录使用vscode配置opencv的全过程&#xff0c;在整个过程中需要用到的工具包括vscode安装包、MinGW-w64和opencv的源码。vs studio配置opencv比较简单&#xff0c;opencv官网中已经有用vs studio编译器编译好的opencv库&#xff0c;但是对于vscode而言&a…

windows下 C++ openCV配置及x86编译(傻瓜式教程)

本傻瓜教程需要的环境如下: IDE: vs2015或vs2017 , windows 10 或 11 vs2017下载地址如下: ①百度网盘 链接&#xff1a;https://pan.baidu.com/s/1r628e9M5lv_F9IWO-h05jA 提取码&#xff1a;23a7 ②官网地址 https://my.visualstudio.com/Downloads/Featured?mktzh-cnh…

【Qt+OpenCV配置简介】

【Qt&OpenCV】QtOpenCV配置简介 文章目录 【Qt&OpenCV】QtOpenCV配置简介前言一、Qt安装二、OpenCV安装三、Qt配置OpenCV四、测试​总结 前言 越来越多的开发人员选择基于开源的Qt框架与OpenCV来实现界面和算法&#xff0c;其原因不单单是无版权问题&#xff0c;更多是…

Qt+OpenCV配置教程(图解亲测)

文章目录 QtOpenCV配置教程安装配置使用 QtOpenCV配置教程 安装 我都安的最新版的&#xff08;cmake 3.22.1 opencv 4.5.1 qt 5.12.1&#xff09;。 1、cmake安装 2、qt mingw安装 3、opencv安装 配置 首先要设置环境变量。 换成对应你自己版本、位置的环境变量 D:\Prog…

Python配置OpenCV

pip install opencv-pyton&#xff0c;但是由于网络原因&#xff0c;会导致下载不成功。可以从这个网站下载python版本对应的opencv安装包&#xff1a;https://www.lfd.uci.edu/~gohlke/pythonlibs/#opencv 比如我的python版本是3.8.8 > 对应python版本&#xff0c;下载这个…