也谈TDD,以及三层架构、设计模式、ORM……:没有免费的午餐
想在园子里写点东西已经很久了,但一直没有落笔,忙着做 一起帮 的开发直播,还有些软文做推广,还要做奶爸带孩子,还要……好吧,我承认,真正的原因是:
太特么的难写了!
但再难写也要写啊,要等到“能写好了再写”,怕是黄花菜都凉了——尤其是技术类文章,时效性非常强的。
刚好坛子里这篇博客:关于拒绝测试驱动开发(NoTDD),看评论争议不小,而这个问题也是我最想写的,所以,蹭个热点,呵呵。
其实我很好奇,博客下面热烈讨论的童鞋,有多少人是真正的在项目中坚持过TDD的。
我公司里的项目,从来没有哪一个项目是要求TDD、能够TDD的;我自己的项目,坚持过TDD一段时间,而且应该是非常久的一段时间,尤其是Entity部分,但现在我基本上都已经放弃了。
为什么呢?
可以洋洋洒洒千言万语,也可以简简单单三个字:不划算。
其实不仅仅是TDD,还包括三层架构、设计模式、ORM等等这些东西,存在大量的争论,莫衷一是:说它好的把它捧到了天上去,说它不行的批得它体无完肤,双方都有大牛为其站台,都可以一二三四五的列出长长的清单,而且每一条都很有道理……
当讨论变成了一种辩论,当辩论变成了一种骂战,最后拼的就是谁的态度更坚决,谁的言辞更犀利,谁的声音更大……所以双方的观点更加的偏激、对立,而这其实无助于我们客观冷静的来分析问题。
说理太枯燥了点,还是听飞哥讲故事吧,呵呵。
最早,我刚接触“设计模式”。什么玩意儿啊?!整本书,就一个感觉,“脱了裤子放屁”。明明一个对象,new一下不就OK了么?什么Factory啊Builder啊,搞毛线呢?所以一直是云里雾里的,包括那些开闭原则、依赖倒置,都似懂非懂的,没帮上我什么忙。
直到有一天,也不知是在哪里,我看到了三个字“上下文”,或者说一句话,大意是:只有理解了上下文,理解了设计模式想要解决的“问题”,你才能真正的理解设计模式。不知道是不是那时候积累也差不多了,茅塞顿开恍然大悟!
我在架构之路(一):目标里说过:设计模式是药,看评论其实很多同学没有理解,对照这句话看能不能明白过来:理解了设计模式想要解决的“问题”……?要解决的问题就是“病”,没病就不要乱吃药;同理,没有“问题”,你也不要乱用设计模式。
一通百通。
所以从最基础的面向对象、到三层架构、ORM、以及敏捷开发、TDD……,所有这些概念方法,本质上都是要解决问题的,而且基本上也是能够解决问题的。
而你,认为它“没用”,其实最大的原因是:你还没碰到这方面的问题。
在这里,大家一定要区分两个概念:“它解决不了问题”和“它(对我)没用”。还是用药做比喻,“这药治不了病”,和“这药(对我)没用”是两个概念。而且尤其要注意的是这两个字:对我。
换到项目中,就是这种架构这种开发模式,适不适合这个项目,能不能解决这个项目开发中遇到的问题。
其实之前我也看到过类似的提法,比如:xxx适合“大”项目。但用“大”和“小”来区分项目,毛糙了一些,很多时候,并不见得正确。最正确的做法是:你了解项目的特点,同时也了解各种模式的优劣,从而能够正确的匹配和选择。当然,这是一个非常庞大的话题,这里没办法展开了。
好,上面我们提到了“优劣”,所谓优势和劣势,但其实,这个提法并不准确。优势,大家都可以承认,解决了问题嘛;但劣势……什么叫做劣势?不服……
我更愿意用另一个词:成本。
“天下没有免费的午餐”。
这是一个经济学上的谚语。一提到这话,我就想起我大学的时候坐在教室里听老师讲《西方经济学》……往事历历在目,谁曾想,我会是今天这个样子?
再说点题外话吧。
【野生程序员】:优先招聘是意气之作,但并非完全意气用事,在我该不该转行?(一)野生程序员的优势一文里,我就较为详细的阐述了野生程序员的优势。简单的说:做架构,做项目管理,需要一个更宏大的视野,而不仅仅是二进制和计算机原理。
这里,我们还是回过头来看,什么叫做“天下没有免费的午餐”?不要理解为“做人不要贪心以免上当”之类的哟!你可以理解为:做任何事情都需要成本。但我更喜欢另一种说法:凡是选择,必有代价。
具体到项目中,不管(注意是不管,无论,随便……)你选择是不是遵循TDD的规范要求,只要你选择了,就必然有代价:
- 不使用TDD,就会在代码的重构、维护、健壮性等方面付出代价;
- 使用TDD,就会在测试代码的开发和维护上付出额外的代价。
无论你怎么选,一定是要付出“代价”的。换言之,代码的“低耦合”“可测试”“便于重构”……不可能从天上掉下来,一定是有成本的!
这本来是一个最简单不过的道理。
然而,当我们迫切的想达到一种目标——尤其是这种目标是美妙的、神圣的、寄托了我们某种强烈情感的时候,我们常常会忘记达成这个目标的成本。
就个人而言,就是通宵达旦废寝忘食乐此不疲,这是你自个儿的事;但对于团队,对于项目呢?“不计一切代价”就是一种蛮干就是瞎搞,后果往往是灾难性……
另一个很有意思的现象:我们的舆论,我们的文化,是鼓励“不惜一切代价”是鼓励“克服重重困难”的,这会让我们有一种莫名的冲动、一种热血沸腾的快感。理智和感性天然就是不兼容的?
那么,我是反对TDD的?
如果你心里还有这样的想法,说明你还是没弄明白我在说什么。
无所谓支持和反对,没有这样简单化的答案。
事实上,你需要的,是做一个成本和收益的分析:针对特定的、具体的项目!
没有一个放之四海而皆准的准则。
不同的项目,有不同的要求,应该因地制宜的采取相应的策略。
这样谈下去还是会很空,我以 一起帮 为例。
我为什么要放弃TDD?因为我对这个项目没有太大的信心,我目前最需要的,是尽快的把项目的原型拿出来,放到市场上进行检验:大家喜不喜欢,有没有前景,收集正面的反面的意见反馈……如果大致符合预期,我就继续做下去;否则,就要快速的进行调整。而我现在的人手又非常有限,好吧,其实就我一个人,所有的代码都得我一个人写;好在网站出bug问题不是很大,所有的用户都是种子用户,他们可以直接的给我反馈而不会因为一两个bug离我而去……
所以综合上面种种考虑,我并不需要TDD,至少暂时不需要。也就是说,代码质量差一点就差一点,可以忍受。如果项目击中了用户的痛点,我可以以后花更大的代价来“补”;如果项目针对的是一个“伪需求”,我就应该尽快止损。
你看,并不是TDD不好,并不是TDD没用,而是我现在“用不着”——这才是三观最“正”的,最无懈可击的理由。·
顺便说一下,我现在采取的策略,我把它称之为“懒人策略”:一开始不写unit test,但一旦出现bug,fix bug之前,首先写unit test,然后在fix。(惭愧啊,仔细想想,这一点我都没完全做到,(⊙﹏⊙)b)
其实我觉得呀,当然仅仅是“觉得”了,大多数的“大牛”们,其实是明白这一点的——虽然他们从没有像我这样系统明确的表述出来。
我这样推断的原因是:现实中确实没有太多TDD实践的项目。
实践TDD的机会其实是非常渺茫的,就我目前能想到的:
- 开发团队,尤其是架构师必须有相当的水平。我在架构之路(三) 单元测试就讲过:单元测试不是那么好写的!凡是可(易于)测试的代码,一定是“低耦合”的,模块之间是具有相当大的“独立性”的,不然相互牵连,将非常难以测试。而随着业务逻辑的耦合度(复杂度)越来越高,解耦的难度也就越来越高。反正据我的观察,一般的开发团队根本hold不住。有时候想想,非常之诡异:耦合度不高的项目,其实又没有多大的必要做TDD?
- 项目负责人对项目能够长期存活具有强大的信心。TDD的实践,是前期投资,后期收获。相当长一段时间,你都会觉得写单元测试非常无聊,只有到了后期,业务逻辑越来越复杂,到处都是千丝万缕的联系,牵一发而动全身,经常一改动单元测试就跑不过的时候,你才会觉得“咦?这玩意还真的有用呢!”但是,注意这个但是,项目负责人有没有足够的信心:这个项目能撑到那个时候?!市场朝秦暮楚变化无常,几乎所有人都是走一步看一步,摸着石头过河,哪里能顾得那么长远?
- 项目从一开始就不赶工期,允许使用大量(至少是双倍)的时间来写单元测试。就算是我有信心这个项目没问题,但时间允许不允许?商场上争的就是一个先手,快鱼吃慢鱼,要快,要抢先占领阵地。这就和强行军一样,确实有很多问题,不如步步为营稳妥,没有重武器,会有掉队减员,部队非常虚弱……但只要先到达阵地,其他一切都在所不惜。
所以,我非常好奇,究竟有多少童鞋真正参与过一个严格按TDD模式实施项目?
那么,TDD是不是就不值得学习了呢?
当然不是的!
+++++++++++++++++++++
真的顶不住了!
12点了,超级 =_=
展开写还有很长很长,强写脑力也跟不上了。先这样吧,有时间我们下次再聊,晚安,各位。
呵呵,偶然中发现的,小小的一个成就,纪念一下。
也谈TDD,以及三层架构、设计模式、ORM……:没有免费的午餐的更多相关文章
- 也谈TDD,以及三层架构、设计模式、ORM……没有免费的午餐,选择了,必付出代价
想在园子里写点东西已经很久了,但一直没有落笔,忙着做 一起帮 的开发直播,还有些软文做推广,还要做奶爸带孩子,还要……好吧,我承认,真正的原因是: 太特么的难写了! 但再难写也要写啊,要等到“能写好了 ...
- 浅谈Java开发三层架构
三层架构,一般来说就是将整个业务应用划分为:表现层(UI).业务逻辑层(BLL).数据访问层(DAL).区分层次的目的即为了“高内聚,低耦合”的思想. 概念简介 1.表现层(UI):简单来说,就是展现 ...
- 【转】浅谈MVC与三层架构
首先给大家引入下MVC的概念: MVC(Model View Controller)模型.视图以及控制器,它是一种较为广泛应用的结构设计模式. 模型:就是在MVC设计模式中需要被显示的数据.在通常情况 ...
- 浅谈.NET,C#三层架构
三层架构 常见架构: 三层(经典) MVC MVVM MVP 开发中常见的23种设计模式: 创建型模式,共五种:工厂方法模式.抽象工厂模式.单例模式.建造者模式.原型模式. 结构型模式,共七种: ...
- 浅谈.NET,C#三层架构(自己总结)
三层架构 常见架构: 三层(经典) MVC MVVM MVP 开发中常见的23种设计模式: 创建型模式,共五种:工厂方法模式.抽象工厂模式.单例模式.建造者模式.原型模式. 结构型模式,共七种: ...
- 关于三层架构与MVC的一些理解
刚毕业的时候,参与了一个上位机的系统开发.上位机所使用的是.net Windows Form技术. 当时,和一个北理的姑娘在一个项目组里.因为她来公司时间比较长,而且经验比较丰富,所以,上位机的架构由 ...
- 浅谈C++三层架构
浅谈C++三层架构 三层架构并不是MVC,MVC是一个很早就有的经典的程序设计模式,M-V-C分为三层,M(Model)-V(View)-C(Control). web开发中的三层架构是指:数据访问层 ...
- JavaEE 三层架构的浅谈
三层架构 三层架构(3-tier architecture) 通常意义上的三层架构就是将整个业务应用划分为:表现层(UI).业务逻辑层(BLL).数据访问层(DAL).区分层次的目的即为了“高内聚,低 ...
- 利用Dapper ORM搭建三层架构
利用Dapper关系对象映射器写的简单的三层架构.Dapper:StackOverFlow在使用的一个微型的ORM,框架整体效率较高,轻量级的ORM框架.网上有较多的扩展.此处只是简单的调用Dappe ...
随机推荐
- .Net Core 连输入中文都变坑了...
前不久Core诞生时候,那个时候我也在项目上没时间去尝那青涩的味道.今天刚刚装上2017就等不及的试了一下. 先创建了一个控制台的应用程序,然后在Main()方法中写了几句话,就等不及的Ctrl+F5 ...
- Centos5搭建vsftpd服务
更换镜像源 由于centos5已经历史久远,内置的镜像源已经不能用.看: 因此,我手工更换了阿里云的源.(ps:我本来是想用网易的源,但不知为什么,这个源在安装vsftpd时提示http 404错误) ...
- ListView的使用(二)长按弹出上下文菜单
public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView ...
- 007---Hibernate基本映射标签和属性介绍
一.映射文件的基本结构举例: <?xml version="1.0"?> <!DOCTYPE hibernate-mapping PUBLIC "-// ...
- JSONObject简介(2)
JSONObject简介 本节摘要:之前对JSON做了一次简单的介 绍,并把JSON和XML做了一个简单的比较:那么,我就在想,如果是一个json格式的字符串传到后台,需要怎么对其处理?如果前台页面需 ...
- 电商app开发新趋势!如何突显竞争力?
2017年是电商变化最大的一年,同时,也是最多机遇的一年,更是电商最好的时代,如最近所看到的亚马逊的市值已经超过了美国8大零售商的总和,带领美国率先走向了新零售时代;马云也在做改变,试图与线下的大卖场 ...
- MyEclipse使用技巧详解
MyEclipse使用技巧的掌握是和我们开发效率挂钩的,那么如何掌握MyEclipse使用技巧呢?这里向你详细介绍了几种使用技巧的操作方法. 在了解MyEclipse使用技巧之前我们来看看MyEcli ...
- Spring读取xml配置文件的原理与实现
本篇博文的目录: 一:前言 二:spring的配置文件 三:依赖的第三方库.使用技术.代码布局 四:Document实现 五:获取Element的实现 六:解析Element元素 七:Bean创造器 ...
- C#码农的大数据之路 - 使用C#编写MR作业
系列目录 写在前面 从Hadoop出现至今,大数据几乎就是Java平台专属一般.虽然Hadoop或Spark也提供了接口可以与其他语言一起使用,但作为基于JVM运行的框架,Java系语言有着天生优势. ...
- Python函数之简单总结
函数的定义 Python的函数定义使用关键字def,如定义一个返回绝对值的函数my_abs def my_abs(x): if x>=0: return x else: return -x 函数 ...