- 浏览: 604768 次
- 性别:
- 来自: 深圳
文章分类
- 全部博客 (300)
- Web前端 (27)
- Java疑难 (60)
- 面试 (59)
- 汇编语言 (1)
- 计算机组成原理 (2)
- 操作系统 (3)
- 计算机网络 (6)
- C/C++疑难 (9)
- SSH (2)
- Web开发 (15)
- 故障 (3)
- 软件开发 (16)
- Portal开发 (1)
- 后台开发 (6)
- 数据库 (35)
- 设计模式 (4)
- 数据结构与算法 (4)
- Linux (3)
- 项目管理 (10)
- 多线程 (12)
- 嵌入式 (9)
- 网络编程 (4)
- 架构设计 (20)
- 软件工具技巧 (7)
- 并行并发 (4)
- 窗口编程 (7)
- 串口编程 (0)
- Flex (2)
- 协议 (1)
- 通讯方式 (4)
- 性能分析优化 (5)
- 测试相关 (4)
- 海量数据处理 (2)
- JAVA核心技术 (10)
- SOA (3)
- 攻略规划 (1)
- 爬虫/搜索 (2)
- 正则表达式 (1)
- A Comparison Of NoSQL Database Management Systems And Models (1)
最新评论
-
charles751:
分析的很好!但有一点:只要同步组合操作就可以了,不一定非要sy ...
Vector 是线程安全的? -
S346618898:
core Java中有一段:Vector类对自己的所有可修改方 ...
Vector 是线程安全的? -
code_cj:
基本上明白了.但执行顺是否应该是/etc/profile -& ...
profile bashrc bash_profile之间的区别和联系 -
xd2008ck:
各自有各自的场景吧楼主不要太激进了
Vector 是线程安全的? -
zwt2001267:
写的不错,赞一个
Vector 是线程安全的?
测试方法存在几个问题:
l 如果测试不够详尽,那么 bug 就会遗留在代码中并潜在地造成严重的问题。
l 测试常常在所有代码编写完毕后编写,难以回头处理程序中的一些问题。
l 测试经常不是由编写代码的程序员编写,所以漏掉一些重要的测试时很有可能的。
l 如果测试编写人员依赖文档或其他东西而不是代码的话,当这些材料与代码不一致就会造成问题。
l 如果测试不是自动进行,它们极有可能不回被经常性地运行。
l 传统的纠正错误的方法极有可能在别的地方引入错误。
测试驱动开发解决了所有这些问题,还附带解决了其他一些问题:
l 在测试驱动开发中,由程序员来完成测试,在代码还在脑海中清晰可见的时候,对测试进行操作。代码是基于测试编写的,这保证了代码可测试性 ,有助于确保测试覆盖的完备性 以及代码与测试的一致性 。所有测试是自动的。我们频繁地以完全相同的方式运行测试。
l 全面彻底的测试覆盖意味着如果在调试阶段引入某个 bug ,测试集就能立刻发现并查明其位置。测试—调试周期就会被控制在相当短的时间内。
l 当系统发布时,详尽的测试集与其一同发布,从而使得将来对程序的修改和扩展更加容易。
获得的是测试得更彻底的代码。你的设计会更加简单,系统(自己)能够清晰地表达你的意图。测试本身就有助于对系统进行描述 。你获得的是瑕疵率极低的系统,而且从始至终都是健壮的。
什么是测试驱动开发?
测试 驱动开发( Test-Driven Development , TDD )是一种开发方式:
l 你要维护一套详尽的程序员测试集 。
l 除非存在相关测试,否则不编写任何产品代码。
l 首先编写测试 。
l 由测试决定需要编写怎样的代码。
编写单元测试是为了测试你所编写的代码是否能够工作,而编写程序员测试是为了定义代码工作的含义 。
极限编程的原则之一就是在与之相关的一套测试还没有被编写出来之前,不要编写任何功能代码。这样做的原因是系统中一切都必须是可测试的 。
先编写少量的测试,随后编写足够使其测试通过的代码,然后再多编写一些测试,接着再编写一些代码,编写测试,编写代码,编写测试,编写代码 ,等等以此类推。
只编写足够让测试通过的代码,编写最简单的但又能工作 的代码。
什么是重构?
对如何做而不是做什么进行修改。
何时进行重构?
l 当存在重复 的时候。
l 当我们觉得代码或代码所表达的意图不清楚 的时候。
l 当我们察觉代码有味儿 ( code smells )的时候,即存在某种微妙的表明可能存在问题的迹象。
把重复的代码抽取到一个独立的方法中。如果重复出现在继承层次中,我们或许能够将重复代码推送到更高一级 的层次上。如果出现重复的是某些代码的结构而不是具体细节,我们可以将不同的部分抽取 出来,并把公共结构部分制作成模板方法 。
采用测试驱动开发能够帮助我们清晰地表达自己的意图,因为在编写代码的时候,我们强迫自己考虑类的接口而不是其实现 。我们有机会站在这个类的用户的立场 上来判定什么才是最有意义的 ,而不是陷入具体的实现细节中去。
如果你觉得有必要编写一条注释的话,请首先考虑重构或重写 代码。注释传达的是为什么这样做 而不是怎么做。
一个类操作另一个类实例,需要把这两者加以合并,要么把实例移动到该类中 ,要么把行为移动到该类中 。一个类对另一个类的内部细节知道的越少越好。让那些彼此了解的代码处于同一个位置。
class Point
{
public :
Point(int x, int y);
int GetX() const ;
int GetY() const ;
void SetX(int x);
void SetY(int y);
void Translate(int dX, int dY);
private :
int _x, _y;
};
class Shape
{
public :
Shape();
void Translate(int dX, int dY);
private :
Point _center;
};
void Shape::Translate(int dX, int dY)
{
/*_center.SetX(_center.GetX() + dX);
_center.SetX(_center.GetY() + dY);*/
_center.Translate(dX, dY);/ / 降低耦合度
}
继承中也会同样的问题,子类过多地了解其祖先类的实现细节,超出了他们应该了解的。可以通过把继承换成委派或将祖先类的具体细节置为私有 的而使这种关系松散化,即去耦。
类的尺寸过大,可以提取出子类 并且采用多态 。方法最好不能超过10 行 左右。当判断较多时可以使用多态 。
如何进行重构?
提取类
当类太大或其行为逻辑组织分散 时,将其切分成多块内聚的行为并在需要的情况下创建新类。把某组行为抽取到一个新类中去。需要某种行为的多重实现时,把易变的代码拆分到一个独立的类中 ,从这个类中提取接口,分别编写所要求的实现。
提取接口
对具体的实现进行抽象,以便更容易地使用一种称作模拟对象(Mock Objectz )的技术。
提取方法
当方法太长或逻辑过于复杂而不易理解时,将其中某些部分提取出来而形成各自独立的方法,将每种不同的功能代码拆分到各自的方法中 。
用子类来代替类型代码
对每种类型分别设计子类,避免复杂的条件判断和switch 语句。
用多态来代替条件判断
形成模板方法
在多个类中都有某种具有相同结构但不同细节的相似方法时,把这个具有相同结构的方法安置在超类中,并把差异化代码提取到独立的方法中并在子类中具体实现 。多态 会负责调用恰当的具体方法。
引入解释变量
把复杂的表达式切分成较为简单的片段,并使用命名良好的变量来更好地传递其中的意图。
使用工厂方法来代替构造方法
所有构造方法具有相同的名字,容易让人搞不清楚,对此我们给每个这样的方法起一个有意义的名字 ,换成工厂方法(静态方法 ),并将构造方法设置成私有的。
Class Rating
{
public :
Rating(int value)
{
Rating(value, “ Anonymous ” , “” );
}
Rating(int value, string source)
{
Rating(value, source, “” );
}
Rating(int value, string source, string review)
{
_value = value;
_source = source;
_review = review;
}
private :
int _value;
string _source;
string _review;
};
// 工厂方法
class Rating
{
public :
static Rating newAnoymousRating(int value)
{
Rating(value, “ Anonymous ” , “” );
}
static Rating newRating(int value, string source)
{
Rating(value, source, “” );
}
static Rating newReview(int value, string source, string review)
{
Rating(value, source, review);
}
private :
Rating(int aRating, string aRatingSource, string aReivew);
};
使用委托来代替继承
只有当子类是特殊种类的超类,或子类对超类进行扩展 而不仅仅是覆写超类的部分功能时,才使用继承。
使用符号常量来代替魔幻数字
使用命名良好的符号常量,需要改动时,到一个地方修改就行了。
使用卫述句来代替嵌套的条件判断
卫述句执行一种非常简单的动作,典型的是return 语句。
Int fib(int i)
{
int result;
if (0 == i)
{
result = 0;
}
else if (I <= 2)
{
result = 1;
}
else
{
result = fib(I – 1) + fib(I – 2);
}
return result;
}
int fib(int i)
{
if (0 == i) return 0;
if (I <= 2) return 1;
return fib(I – 1) + fib(I – 2);
}
意图导向的编程
名字
使用名词或名词短语作为类的名字
使用形容词或具有一般性的名词或名词短语来为接口命名
通常以 -able 来结尾,避免用字母 ”I” 开头或结尾。
使用动词或动词短语作为方法名
使用公认的取值方法和赋值方法的命名习惯
如 getX 和 setX 来取得或修改一个名为 x 的变量。当你想要的值是对象的属性而不是某一项特征时往往去掉 get 前缀更清楚,如 size() 而不是 getSize() 。
不要在方法名中保留多余的信息
如果一个用来增加 X 实例的方法,一般会为这方法取名为 add 而不是 addX ,因为这个方法的参数就是一个 X 。这样做的好处是如果将来重构改变了参数的类型,方法名也不会与其参数脱节 。
使用名词或名词短语作为变量名
简单
怎么简单怎么来
通过重构来使代码保持简单
用做有根据的假设
在我们编写测试(以及后来编写真实代码)的时候做些有意义而且可读性强的假设。
“不要注释”与正当的注释
发表评论
-
Maven实战(三)——多模块项目的POM重构
2015-05-20 15:28 871转自http://www.infoq.com/cn/news ... -
java常用工具
2015-03-30 17:05 788jpsjstatjstackjinfojmap -
Spring配置文件xsi:schemaLocation无法解析导致启动失败的解决方案
2014-03-27 16:59 3037来源http://www.jnan.org/archives ... -
抽象类与接口选择
2013-12-06 00:20 910从设计理念层面看 abst ... -
各种流行的编程风格
2013-09-10 23:27 749转自http://coolshell.cn/arti ... -
字符集与字符编码
2013-09-06 00:23 1291摘自http://blog.sina.com.cn ... -
由12306.cn谈谈网站性能技术
2013-08-28 23:57 872转载自http://coolshell.cn/article ... -
webservice通信原理
2013-05-13 11:17 1105当前,WebService是一个热 ... -
java序列化(Serializable)的作用和反序列化
2013-04-23 15:23 9091、序列化是干什么的? 简单说就是为了保存在内存中的各种对象的 ... -
深入Java事务的原理与应用
2013-04-23 14:57 893一、什么是JAVA事务 通常的观念认为,事务仅与数据库 ... -
Session机制详解
2013-04-23 14:53 1055http://tech.it168.com/j/2006-07 ... -
JTA 深度历险 – 原理与实现
2013-04-23 14:46 767简介: 在 J2EE 应用中,事务是一个不可或缺的组件模型, ... -
Java程序性能优化之找出内存溢出元凶
2013-04-07 00:30 928我曾经在刚入行的时候做过一个小的swing程序,用到了jav ... -
用exe4j将java程序做成exe程序
2013-01-31 00:35 1102步骤如下: 1.在桌面新建一个文件夹,如AnswerSy ... -
调用WebService服务客户端代码编写
2011-07-01 21:05 11151调用WebService服务客户端代码编写 目前比较 ...
相关推荐
测试驱动开发(Test-Driven Development,简称TDD)是一种敏捷软件开发的技术,以测试作为开发过程的中心环节。它倡导在编写产品代码之前先编写测试代码,确保产品代码能够通过这些测试。这种方法与传统的开发方式...
测试驱动开发实践介绍 测试驱动开发(Test-Driven Development,TDD)是一种软件开发过程,它强调在编写实际代码之前先编写自动化测试。这种方法可以帮助开发者编写更好的代码,提高代码质量和可维护性。 在测试...
《Java测试驱动开发》介绍如何将各种TDDzui佳实践应用于Java开发,主要内容包括:用Java语言进行TDD会用到的各种工具和框架,所需环境搭建;通过实际应用程序,展示TDD优点及开发中应注意的主要问题;TDD是如何通过...
要使测试驱动开发在软件行业中得以繁荣兴盛,需要一些条件,《C#测试驱动开发》从讨论这些条件开始。软件开发发展到今天,有其历史和特定的条件,理解这些很重要。避免重复过去的错误也很重要。在自己当前的开发实践...
极限编程反其道而行之,主张采用测试驱动开发(TDD)的方法,即通过测试定义所要开发的功能的接口,然后实现功能的开发过程。TDD通过不断地测试推动代码的开发,既简化了代码,又保证了软件质量。本书采用“手把手”...
《测试驱动开发》是Kent Beck的经典著作,这本书深入探讨了测试驱动开发(TDD)这一软件开发实践。TDD是一种编程方法论,它强调在编写实际功能代码之前,先编写测试用例,以此来指导软件设计和编码过程。通过这种...
### 测试驱动开发(TDD)概述 测试驱动开发(Test-Driven Development,简称TDD)是一种软件开发方法论,它要求在编写实际功能代码之前先编写测试用例。这种方法有助于确保代码的质量,并使得代码更加健壮、易于维护...
测试驱动开发(Test-Driven Development,简称TDD)是一种软件开发方法,它的核心思想是先编写测试用例,再编写满足这些测试用例的代码。这种方法强调在编码之前,先明确需求并创建能够验证功能是否正确的测试。TDD...
测试驱动开发(Test-Driven Development,简称TDD)是一种软件开发方法,由Kent Beck在其著作《测试驱动开发:通过实例》中提出并详尽阐述。这种方法主张先编写自动化测试用例,然后再编写满足这些测试的代码,从而...
花井志生*的《C现代编程(集成开发环境设计模 式*限编程测试驱动开发重构持续集成)》从使用C语 言进行嵌入式开发的特点入手,主要讲解了如何将集 成开发环境、设计模式、*限编程、测试驱动开发、 重构、持续集成这些...
测试驱动开发(Test-Driven Development,简称TDD)是一种软件开发方法,由Kent Beck在其同名著作《测试驱动开发》中提出。这种方法主张先编写测试用例,再编写实现功能的代码,以此来驱动开发过程,确保代码的质量...
《java测试驱动开发》 源码 本书介绍如何将各种TDD最佳实践应用于Java开发,主要内容包括:用Java语言进行TDD会用到的各种工具和框架,所需环境搭建;通过实际应用程序,展示TDD优点及开发中应注意的主要问题;TDD是...
测试驱动开发(TDD)是极限编程的重要特点,它以不断的测试推动代码的开发,既简化了代码,又保证了软件质量。本文从开发人员使用的角度,介绍了 TDD 优势、原理、过程、原则、测试技术、Tips 等方面。 背景 一个...
### Python测试驱动开发 #### 知识点概览 1. **测试驱动开发(TDD)的概念** - 定义与原则 - TDD在软件开发生命周期中的作用 - 实施TDD的好处与挑战 2. **Python与测试驱动开发** - Python作为TDD的理想语言 -...
测试驱动开发(TDD)是一种敏捷软件开发技术,它要求开发者在编写功能代码之前先编写测试代码。这种方法提倡先写失败的单元测试,然后编写刚好足够使测试通过的代码,最后通过重构来提高代码的质量。王晓毅所著的...
《Python测试驱动开发:使用Django、Selenium和JavaScript进行Web编程(第2版)》是一本详尽探讨如何在Web开发中应用测试驱动开发(TDD)技术的专业书籍。这本书不仅涵盖了Python语言的基础,还深入讲解了Django框架...