10/23日,我在部门内部进行了一次内部学习,使用目前流行的Coding Dojo(道场)方式,进行了TDD开发的演练。演练的题目如下:
 
 
有关Coding道场的介绍,请自行百度一下,我就不再多做介绍了。
 
从效果来看,基本达到了传达TDD是什么样的开发方式的目的。尤其是大家从最初满脑子如何实现这个程序,怎样去设计算法,逐渐转变为了先想如何测试,从最简单的实现开始,最终演化成最终的设计。当然,目前为止,参加人员也只是理解了TDD是一个什么样的开发方式而已,还谈不到真正使用TDD进行开发。这需要一个更加长期的自我训练和使用的过程。使用TDD,最主要的是一种思维方式的变化。
首先:要坚信所有的程序皆可测,如果不能测试,不是产品的特性导致,而是自己的能力不足导致,设计上有问题。因此必须从设计上加以改变,使得程序可测。如一般认为曲线的显示是否正常,是无法使用自动测试的。换个角度:如果显示只是一个数据-坐标的转换的话,测试的重点就变成了数据是否正确,而这一点是完全可测的。
其次:虽然需要全局的考虑,但是要从简单入手,演进式设计。
这一点,在本次道场演练中体现的就很明显,此次道场开始,很多人的想法就是,建立某种算法,将需要的字符显示出来。于是第一个函数就是:void DisplaySegmentDigital(String input),然后再写那些子函数。如何测试这个函数?这是一个输出到屏幕显示的函数,它只能用眼来判断,显然不适合自动测试或者单元测试。所以,TDD不是一个先实现框架,再实现具体功能的做法。输出到屏幕,只是最后的一个过程,也是一个简单的过程,因此可以不必作为重点。重点在于显示的数据是什么?所以,函数就变为了:String DisplaySegmentDigital(String input)。这个时候,这个函数不再是向屏幕输出,而是输出一个字符串,再由另外一个字符串显示函数完成向屏幕的输出。而原来这个函数就变得可测了。于是,第一个测试函数被写出来了:
String strText = "910"; String strTextResult = "._.|_|..|.....|..|._.|.||_|";  
String strOutput = digitalSegment.DisplaySegmentDigital(strText);   
assertEquals(strOutput , strTextResult);
第一个测试顺利通过,因为实现非常简单:
public String DisplaySegmentDigital(String strText) {
    return "._.|_|..|.....|..|._.|.||_|";
}
接下来的困难是:下一个测试什么?测试“3456”的输出?OK,我们先试试看,于是我们想写第二个测试:

String strText = "3456"; String strTextResult = "????????";

问题接着出来了:这串问号该填什么?这样测试真的有意义么?几乎所有的人都直觉得发现这里有问题。简短的讨论后,结论是应该测试每个数字的显示,而非一个字符串。于是,测试变为:

String strText = "9"; String strTextResult = "._.|_|..|";

String strOutput = digitalSegment.DisplaySegmentDigital(strText);

assertEquals(strOutput , strTextResult);

实现变为:

public String DisplaySegmentDigital(String strText) {

if(strText == "9")

return "._.|_|..|";

else

return null;

}

实现后,接着测试:

strText = "1"; strTextResult = ".....|..|";

strOutput = digitalSegment.DisplaySegmentDigital(strText);

assertEquals(strOutput , strTextResult);

实现也变为:

public String DisplaySegmentDigital(String strText) {

String[] strResult=new String[10];

strResult[0]="._.|.||_|";

strResult[1]=".....|..|";

strResult[9]="._.|_|..|";

return strResult[Integer.parseInt(strText)];

}

至此,很显然我们的算法也就自然而然的诞生了。可能与很多人自己开始的算法设计不太一样,但也不应该差到哪里:)。这就是TDD演进式设计。

但有个问题,._.|_|..|是什么东东?我怎么知道最终输出是正确的。因此,我们稍微改变了一下写法:

String strText = "9";   String strTextResult = "._." +
                                                                 "|_|" +
                                                                  "..|";

String strOutput = digitalSegment.DisplaySegmentDigital(strText);

assertEquals(strOutput , strTextResult);

strText = "1";

strTextResult = "..." +

"..|" +

"..|";

strOutput = digitalSegment.DisplaySegmentDigital(strText);

assertEquals(strOutput , strTextResult);

实现变为:

public String DisplaySegmentDigital(String strText) {

String[] strResult=new String[10];

strResult[0]="._." +

"|.|" +

"|_|";

strResult[1]="..." +

"..|" +

"..|";

strResult[9]="._." +

"|_|" +

"..|";

return strResult[Integer.parseInt(strText)];

}

现在直观多了。

为什么一定要变得直观,其目的不单纯是为了程序的易读性,更重要的是:测试不应该抄实现的代码,实现也不要抄测试的代码,否则后果很严重。写测试代码时,必须是含着测试的心态,含着使用者的心态去写测试代码,而非一门心思去想实现。如果这样,TDD就失败了。这也是为什么TDD要求先写测试代码,再写实现代码的原因。因为我们一旦先想到了实现,那么接下来的测试,必然会跟着实现的逻辑走,从而违背“测试独立性”的原则。实现发生错误,测试也无法发现。

接下来需要整理一下代码,显然DisplaySegmentDigital这个函数名不是那么准确,后来议论了一番,得出的名字是:GetDigitalDisplayContent。结束后,我想GetDigitalFont可能更好。

好了,有关第一次道场就写到这里,留一个小小的问题:到目前为止我们还没有设计显示的算法。为了便于输出,目前的设计如何改进?

Coding道场:第一次的更多相关文章

  1. Python全栈开发:冒泡排序

    #!/usr/bin/env python # -*- coding;utf-8 -*- """ 第一次对比:找到最大值,放到最后 对比是两两对比,对比的两个数组合共有l ...

  2. 软件工程 Coding.net代码托管平台 Git初学者的使用总结 五步完成 程序,文件,文件夹的Git

    一.前言 第一次用git相关的命令行,我使用的是Coding.net代码托管平台.Coding.net 自主打造的基于 Git 的代码托管平台,提供高性能的远端仓库,还有保护分支,历史版本分屏对比. ...

  3. Coding 及 git 的工程使用方法

        在过去的两周,同学们除了在学习 C 语言之外,还在学习如何利用 git 将自己的代码上传到 coding 中.也有大量的同学,成功的上传了代码.但是,实际上大部分同学的用法都不合理.这里,以一 ...

  4. C 语言学习 第一次作业总结

    第一次的作业是冯老师布置的练习题,需要在pta平台上完成.我这边看不到结果,但是透过冯老师给出的截图,同学们都还是认真的去做的.同时,我这边也布置了一个持续 3 周的作业:熟悉 git 的使用.因为后 ...

  5. Coding编译连接过程中遇到的问题及解决方法(iOS)

    Coding 上下载地址:https://coding.net/u/coding/p/Coding-iOS/git Github源码下载地址:https://github.com/Coding/Cod ...

  6. Coding源码学习第三部分(EaseStartView.m)

    首先接上篇的要做一个NSEnumerator 类的延展阅读. 枚举(NSEnumerator) (1)依附于集合类(NSArray,NSSet,NSDictionary),没有用来创建实例的接口. ( ...

  7. Coding源码学习第一部分(AppDelegate.m)

    前言:在此首先感谢开源,感谢大神们的无私分享. Coding 的主页:https://coding.net/app#app-feature Coding 自己家的仓库:https://coding.n ...

  8. 使用VS2010在Coding.net上进行代码托管

    网上有VS2010和Github结合使用办法,但是Github在国内使用太慢,本文使用相同的配置方法稍作改动让VS2010代码托管在coding.net平台上.由于只是稍做记录让自己不会遗忘,所以叙述 ...

  9. Coding Kata - 挑战你的“底线”

    Coding Kata简介 如何进行Kata练习 亲身感受 Coding Kata简介 前段时间听到一个比较有意思的概念叫做Coding Kata,今天试了一下来说说一些想法和思考.Kata是一个日语 ...

随机推荐

  1. 关于多字节、宽字节、WideCharToMultiByte和MultiByteToWideChar函数的详解

    所谓的短字符,就是用8bit来表示的字符,典型的应用是ASCII码. 而宽字符,顾名思义,就是用16bit表示的字符,典型的有UNICODE. **************************** ...

  2. PHP面试题目搜集

    搜集这些题目是想在学习PHP方面知识有更感性的认识,单纯看书的话会很容易看后就忘记. 曾经看过数据结构.设计模式.HTTP等方面的书籍,但是基本看完后就是看完了,没有然后了,随着时间的推移,也就渐渐忘 ...

  3. 原生JS实现jquery的链式编程。

    这是我根据之前遇到的一个面试题,题目:用原生JS实现$("#ct").on("click",fn).attr("id"). 然后看了篇jqu ...

  4. Cesium原理篇:4Web Workers剖析(2)

    What's the WebWorkers? 2008 年 W3C 制定出第一个 HTML5 草案中提出了工作线程(Web Worker)的概念,并且规范出 Web Worker 的三大主要特征:能够 ...

  5. 原创:跳坑指南——微信小程序真机预览跟本地不同的问题

    微信小程序中出现最多的一个问题,就是真机跟本地不同:我简单列举一些我发现的原因,给大家参考,大家也可以把自己发现的东西回复给我,给我参考:本地看不到数据,就先让本地能看到数据,再看本帖.... 1:本 ...

  6. Angular依赖注入详解

    Angular算是将后端开发工程化引入前端的先驱之一,而Dependency injection依赖注入(后面简称为DI)又是Angular内部运作的核心功能,所以要深入理解Angular有必要先理解 ...

  7. SSE指令集优化学习:双线性插值

    对SSE的学习总算迈出了第一步,用2天时间对双线性插值的代码进行了优化,现将实现的过程梳理以下,算是对这段学习的一个总结. 1. 什么是SSE 说到SSE,首先要弄清楚的一个概念是SIMD(单指令多数 ...

  8. 为你带来灵感的 20 个 HTML5/CSS3 模板

    1. Curve 2. Tapestry 3. Aqueous 4. Deliccio 5. Respond 1.5 6. Triangle Responsive 7. Design Company ...

  9. ASP.NET WEB API必知必会:特性路由

    一.什么是特性路由? 特性路由是指将RouteAttribute或自定义继承自RouteAttribute的特性类标记在控制器或ACTION上,同时指定路由Url字符串,从而实现路由映射,相比之前的通 ...

  10. js树形控件—zTree使用总结

    0 zTree简介 树形控件的使用是应用开发过程中必不可少的.zTree 是一个依靠 jQuery 实现的多功能 “树插件”.优异的性能.灵活的配置.多种功能的组合是 zTree 最大优点. 0.0 ...