为什么从前那些.NET开发者都不写单元测试呢?

楔子

四年前我虽然也写了很多年代码,由于公司虽然规模不小,却并非一家规范化的软件公司,因此在项目中严格意义上来说并没有架构设计、也不写单元测试,后来有幸加入了一家公司,这家公司虽然也是一家小公司,但是好歹曾经聘请过一位架构师,这位架构师使用spring.net 搭建了一套基础的技术架构,并在公司推广使用EnterpriseArchiture(EA)软件设计UML图,但是由于种种原因,他试用期还没过,只是简单的设计了几个业务之后就离职了。

我加入这家公司的时候距离这位架构师离职也已经有相当长的时间,未能直接学习的机会,但是非常还能接触到这位优秀架构师留下来的部分代码和资料,这为我的职场之路打开了新的窗户,从这些资料中学到了许多之前一直想学但是没机会学的内容,并第一次接触到单元测试这种技术。

然而,值得讽刺的是,在他搭建的项目架构中,单元测试代码也只有一个简单的示例,并没有针对业务功能编写任何的具体测试代码,而实际上经手这个项目的开发者已经超过了几十个人,其实这也从一个侧面说明中国相当一部分.Net 开发者跟我一样,几乎都不写单元测试。甚至是不少开源代码贡献者,他们贡献的开源组件,也并没有设计单元测试的代码。但是再来看国外的一些优秀的开源组件,往往也都会提供单元测试的代码,TDD实际上正是一种非常便捷高效门槛低的设计模式。

后来,我经手的每一个项目都会按照TDD的方式写单元测试,但是依然没有深刻领悟单元测试的精髓,只是把它当做代码调试的一个入口而已。其实这种单元测试依然不合格,尤其是认真的运行一下代码覆盖率报告之后,就会发现,单元测试的代码覆盖率只有10个点,与微软的官方标准单元测试覆盖率100%相比,简直就是不合格。

什么是单元测试?

一直以来大家有一种错误的理念,总是认为单元测试作为一种测试方法,主要是由测试工程师来主导的一种测试手段。并非如此,借用教科书上的说法,单元测试作为一种最基础的测试手段,其主要作用是“在计算机编程中,针对程序模块(最软件设计中的最小单元)进行正确性检验的测试工作。”

之前曾经阅读过一篇文章,作者提了一个例子,他说他儿子在使用乐高积木做玩偶过程中,组装完一些零部件之后,就会试图去测试一下部件是否能够正常运转,作者说这种过程实际上就是一种单元测试。他认为,单元测试的思想其实普遍存在于人类文明的发展过程中,从人类开始制作工具来说,就需要使用单元测试的方法对工具的可用性进行测试。尤其是进入工业时代以来,传统制造业尤其重视产品质量,为了提高产品质量,采取了一系列措施,这种例子其实不胜枚举。

去年我曾经有幸为一家国企开发过一些质量管理系统,并深入了解了这些企业的质量管理流程,虽然从宏观层面来看,以互联网经济为代表的新兴经济对以包括制造业在内的实体经济带来了许多冲击,但是依然那些优秀的制造业企业依然坚定的将产品质量作为核心竞争力,并通过互联网的这种模式,让企业的发展获得了新的机遇。

软件行业其实本质上和制造业没有那么巨大的区别,以软件开发过程中,由软件工程师们开发出来的每一个方法实际上就是开发者们解决虚拟世界生存问题的零部件,每一个产品或应用就是这样的工具。在一个软件开发过程中,往往需要涉及到多人操作,经常需要调用别人编写的模块代码。尤其是在面向互联网开发中,如果不能认真的对待每一行代码,就有可能一些疏忽大意造成不可弥补的后果。

因此做好单元测试,不仅仅只是开发者们应该采用的一种测试手段,而是应该是一种基本的技术手段,对我们开发的每一行代码,都应该使用单元测试进行覆盖,自己负责的模块定义应该尽量明确,模块内部的更改不会影响他模块,让模块的质量得到稳定和良好的保证。

为什么大家认为单元测试不合时宜?

当然,大部分开发者也许明白这个道理,但是却认为单元测试不合时宜,主要包括以下几点:

1,没有时间 
---- 在软件开发过程中,往往以完成任务为第一要务,而由于时间计划的安排,单元测试的设计和实践均需要花费大量的时间完成,过度的使用单元测试会拖延项目的进度,而且花费的边界成本高昂而收效甚微。 
---- 笔者认为,跟随代码一起,设计和应用单元测试,并不会显著的带来时间上的损害,反而会让开发者更加关注代码本身的目的,让输入输出和计算过程更为合理。

2,计划会变化:需求变化快,单元测试跟不上需求的变化 
由于软件产品需求的不确定性,导致之前设计的单元测试并非符合项目的实际功能性需求,没必要花费这个精力设计这种没用的场景。 
----笔者认为,需求变化与是否应用单元测试关系不大。

3,单元测试是用来找Bug的,应该有测试来编写 
单元测试作为一种测试手段,是为了发现代码中存在的bug。 
--- 笔者认为:单元测试的目的是为了让程序作者更加快捷的发现代码中存在的问题,并在第一时间发现和解决,从而保障软件质量。

4,有程序员定义的单元测试没有意义,因为程序员设定的逻辑场景就有可能不准确 
单元测试作为由程序员根据实际用例出发设计的测试流程,本身可能并非符合业务的实际应用需要,因此单元测试应该由最了解需求的人来进行设计。而且完成单元测试设计后,也需要有了解需求的人对单元测试用例和代码进行审查。 
--- 笔者认为:由程序员开发的单元测试没有意义,那么他写的代码呢?写代码本身就是为了完成指定的用例场景,如果单元测试都不合理,那么如何确保代码本身就合理?

5、TDD测试驱动,不过是一种形式主义,本身就不合时宜 
在TDD模式中,测试先于开发,所以需要经过良好的设计和定义,最好能够解耦合各个模块,让代码更加完美的匹配测试代码。但是这种代码本身就要求经验丰富的开发者才能完成,而能够完成这种能力的,本身就属于优秀开发者,并不需要设计单元测试代码。 
-- 笔者认为:优秀的开发者,本身不仅仅只是因为他们出身优秀,而是因为他们掌握了包括单元测试等在内的优秀学习方法并每天都在实践。

好的单元测试的标准

笔者最近阅读《构建之美》(第一版)深刻认同提出的关于好的单元测试的标准,例如以下几点: 
1、单元测试应该在最低的功能\参数上验证程序的正确性。 
2、单元测试应该由最熟悉代码的人(程序的作者)来写。 
3、单元测试过后,机器状态保持不变。---笔者认为,不仅仅机器的状态应该保持不变,数据中存储的数据关系也应该保持不变,例如,通过单元测试录入的数据,应该在完成测试后自动回滚,或者添加测试戳记后,手动清除。 
4、单元测试要快(一个测试的运行时间是几秒钟,而不是几分钟)。 
5、单元测试应该产生可重复、一致的结果。 
6、单元测试具有独立性的特点,单元测试的运行、通过、失败不依赖于别的测试,可以认为构造数据,以保持单元测试的独立性。 
7、单元测试应该负载所有的代码路径。 
8、单元测试应该继承到自动测试的框架中。 
9、单元测试必须与产品代码一起保存和维护。

何妨不试一试呢?

在软件企业的发展前期,往往会由于技术上的一些偶发性突破,取得短期的优势,但是随着企业的规模发展,使用更加规范化甚至成为形式化的方式确保产品的质量是必然趋势,而单元测试则是一种看似简单,但却效果显著的手段,相信通过一段时间的实践,就会深刻的领悟到其中的妙处。

由于时间仓促,可能没什么干货,下一篇再进一步讨论单元测试、代码覆盖率的问题。

原文地址:https://www.cnblogs.com/xiyuanMore/p/10618130.html

.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com
640?wx_fmt=jpeg


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/316507.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

git 命令的使用

git 各个命令的详细使用方法参考git-docs。 Table of Contents git Table of Contents add archive blame branch checkout cherry-pick clone commit config diff fetch for-each-ref github init log & show merge mv pull push rebase remote r…

求和(莫比乌斯反演)

由于找不到提交网站,所以不保证正确性哦~ problem 已知积性函数 F(1)1,F(pk)pk1F(1)1,F(p^k)p^k1F(1)1,F(pk)pk1。(ppp为质数,k∈Zk∈Z^k∈Z) 给定 nnn,求 ∑i1nF(i)\sum_{i1}^n F…

P2467 [SDOI2010]地精部落

P2467 [SDOI2010]地精部落 题意: 有n个山脉高度分别是1到n,现在让你按照山峰山谷的顺序依次摆放(第一个可以是山峰也可以是山谷),问有多少方案(答案mod p) 题解: dp,但是自己推不出来 这个博客讲的非常详细&#x…

使用 xUnit 编写 ASP.NET Core 单元测试

还记得 .NET Framework 的 ASP.NET WebForm 吗?那个年代如果要在 Web 层做单元测试简直就是灾难啊。.NET Core 吸取教训,在设计上考虑到了可测试性,就连 ASP.NET Core 这种 Web 或 API 应用要做单元测试也是很方便的。其中面向接口和依赖注入…

[刷题记录] luogu网络流24题 及 网络流心得体会 及 经典模型不定期更新

文章目录信息汇总表格飞行员配对方案问题分配问题运输问题数字梯形问题最小路径覆盖问题魔术球问题圆桌问题试题库问题深海机器人问题航空路线问题火星探险问题太空飞行计划问题方格取数问题骑士共存问题最长不下降子序列问题孤岛营救问题汽车加油行驶问题[CTSC1999]家园 / 星际…

2021牛客暑期多校训练营5

2021牛客暑期多校训练营5 题号题目知识点AAway from CollegeBBoxes概率CCheating and StealingDDouble Strings线性dpEEert EsiwtibFFinding PointsGGreater Integer, Better LCMHHolding Two签到IInterval QueriesJJewels最小权匹配KKing of Range尺取法

记录使用 Cake 进行构建并制作 nuget 包

前段时间折腾了一下,总算是把我自己的图片缓存控件(https://github.com/h82258652/HN.Controls.ImageEx)发布到了 nuget 上,目前已经进入一个比较稳定的版本了,基本没有很严重的 bug 了。其实核心代码早就写完了&#…

2021牛客暑期多校训练营6

题号题目知识点AContracting Convex HullBDefend PonyvilleCDelete EdgesDGambling MonsterEGrowing TreeFHamburger SteakGHasse DiagramHHopping RabbitIIntervals on the RingJDefend Your CountryKStarch Cat

DDD领域驱动设计理论篇 - 学习笔记

一、Why DDD?在加入X公司后,开始了ASP.NET CoreDockerLinux的技术实践,也开始了微服务架构的实践。在微服务的学习中,有一本微软官方出品的《.NET微服务:容器化.NET应用架构指南》是我们学习的葵花宝典,纵观微软官方放…

Intervals on the Ring

Intervals on the Ring 题意: 给出环上的一组区间,需要构造环上的一组区间使得这些区间的交是给定的区间的并 题解: 集合的摩尔定理告诉我们:∪Ai‾∩Ai‾\overline{\cup A_{i}}\cap\overline{A_{i}}∪Ai​​∩Ai​​&#xf…

CodeForces 516E Drazil and His Happy Friends(数学+最短路)

problem 洛谷链接 solution 记 gcd⁡(n,m)d\gcd(n,m)dgcd(n,m)d,则快乐只会在同余 ddd 的关系中传递。 ixn≡ixs⋅d≡i(modd),iym≡iyt⋅d≡i(modd)ixn\equiv ixsd\equiv i\pmod d,iym\equiv iytd\equiv i\pmod dixn≡ixs⋅d≡i(modd),iym≡iyt⋅d≡i(modd)。 所…

.NetCore使用skywalking实现实时性能监控

一、简介很久之前写了一篇 《.Net Core 2.0 InfluxDBGrafanaApp Metrics 实现跨平台的实时性能监控》关于NetCore性能监控的文章,使用InfluxdbAppMetrics进行项目性能监控,由于技术有限,在正式环境使用一段时间后,莫名的AppMetric…

Hamburger Steak

Hamburger Steak 题意: 有m个锅,n块肉,每个肉需要煎ai分钟才能熟,每个锅同时最多煎一块肉,一块肉最多可以被两个锅煎(但不能同时),问最少多长时间能全部煎熟,并输出方案 题解: 我…

「ROI 2017 Day 2」反物质(单调队列优化dp)

problem LOJ2772 solution 这道题求的是“保证最多”,这个“保证”真的屑啊! 因为我们无法确定实验生成的克数,所以我们应当计算的是最坏情况。 又因为我们可以选择做哪些实验,所以要的是所有实验组合最坏情况下的最大价值。…

netcore开发windows普通服务(非Web)并一键发布到服务器

netcore下开发windows服务如果是web项目的话,由于aspnetcore本身是支持的,把默认的host.Run改为host.RunAsService就可以了。但是普通的netcore的控制台项目我终于找到了如下方式来实现:Microsoft.Extensions.HostingSystem.ServiceProcess.S…

Hopping Rabbit

Hopping Rabbit 题意: 给你n个矩阵,每个矩阵(给出左上标和右下标),现在让你给出一个点的位置,这个点每次只能上下左右四个方向移动,且移动距离为d,是否存在一个这样的点,其所有落点都不在矩阵…

删好串(区间dp)

problem 定义长度为 nnn 的“好”的串 aaa 满足: ∣ai−ai−1∣1,i∈[2,n]|a_i-a_{i-1}|1,i\in[2,n]∣ai​−ai−1​∣1,i∈[2,n]。ai≥ai−1ai12,i∈[2,n−1]a_i\ge \frac{a_{i-1}a_{i1}}{2},i\in[2,n-1]ai​≥2ai−1​ai1​​,i∈[2,n−1]。 给定长度为 nnn 的序…

直播预告 - 微软MVP为你揭秘Visual Studio 2019新特性

作为"宇宙第一IDE“的Visual Studio集成开发环境,已经经历了超过十几年的迭代成为一款功能丰富且高效的开发工具,微软自己给Visual Studio 的定位是 “更快、更可靠,对个人和团队更具生产力,更易于使用,并且更容易…

Delete Edges

Delete Edges 题意&#xff1a; 给出一个n个点的完全图&#xff0c;删一些三元环使得边数<n 输出所删的三元环 3<n<2000 题解&#xff1a; 肯定是结论题&#xff0c;但是我不会。。 结论&#xff1a;xyz0(mod n)1<x<y<z<n所有解即可 证明过程可以看看…

.NET 机器学习生态调查

机器学习是一种允许计算机使用现有数据预测未来行为、结果和趋势的数据科学方法。 使用机器学习&#xff0c;计算机可以在未显式编程的情况下进行学习。机器学习的预测可以使得应用和设备更智能。 在线购物时&#xff0c;机器学习基于历史购买推荐你可能喜欢的其他产品。 刷信用…