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. 1Z0-053 争议题目解析686

    1Z0-053 争议题目解析686 考试科目:1Z0-053 题库版本:V13.02 题库中原题为: 686.You execute the following FLASHBACK TABLE com ...

  2. selenium操作隐藏的元素

    有时候我们会碰到一些元素不可见,这个时候selenium就无法对这些元素进行操作了.例如,下面的情况: Python 页面主要通过“display:none”来控制整个下拉框不可见.这个时候如果直接操 ...

  3. struts2学习笔记--使用servletAPI实现ajax的一个小Demo

    这个例子是点击网页上的一个button,然后调用action,使用response项前台打印"哎呦 不错哦",当然是以异步形式实现. jsp页面: <head> < ...

  4. jquery动画,基础以及我发现的新大陆

    $.animate()在jquery官方介绍有2中方式,其实我发现的新大陆也是第二种方式的扩展! 一.$.animate( properties [, duration ] [, easing ] [ ...

  5. 【原创】SQL审核系统

    前言 应公司dba的需求,做了一个sql审核并可以执行.记录的系统,整个系统有三种角色: DBA(需要管理员手动登记) 运维人员(需要管理员手动登记) 开发人员(默认注册后用户) 权限分配如下: 一些 ...

  6. Html 文档在线编辑器

    // //

  7. vue+sass 下sass不能运行问题

    好久没写博文了,今天抽空写一写,最近在用vue.js build 项目,今早想使用sass来编译css,可是安装好依赖包之后仍然显示一下错误: ERROR in ENOENT: no such fil ...

  8. Visual Studio Code 1.0发布,支持中文在内9种语言

    Visual Studio Code 1.0发布,支持中文在内的9种语言:Simplified Chinese, Traditional Chinese, French, German, Italia ...

  9. 基于.net mvc 的供应链管理系统(YB-SCM)开发随笔1-开篇

    作为开篇之作,先把这个项目的介绍和一些技术点给各位. 1.项目所用到的技术 (1)前台展示:ASP.NET MVC 3.0+Jquery+Sea+Bootstrap等 (2)开发环境:VS2012/V ...

  10. Angular2正式版发布,Wijmo抢先支持

    Angular2正式版发布 9月15日,Angular 2 的最终版正式发布了.作为 Angular 1 的全平台继任者 -- Angular 2 的最终版,意味着什么? 意味着稳定性已经得到了大范围 ...