利用Ant和JUnit进行增量开发.doc

上传人:枫** 文档编号:562111672 上传时间:2023-02-14 格式:DOC 页数:12 大小:107KB
返回 下载 相关 举报
利用Ant和JUnit进行增量开发.doc_第1页
第1页 / 共12页
利用Ant和JUnit进行增量开发.doc_第2页
第2页 / 共12页
利用Ant和JUnit进行增量开发.doc_第3页
第3页 / 共12页
利用Ant和JUnit进行增量开发.doc_第4页
第4页 / 共12页
利用Ant和JUnit进行增量开发.doc_第5页
第5页 / 共12页
点击查看更多>>
资源描述

《利用Ant和JUnit进行增量开发.doc》由会员分享,可在线阅读,更多相关《利用Ant和JUnit进行增量开发.doc(12页珍藏版)》请在金锄头文库上搜索。

1、利用Ant和JUnit进行增量开发作者:Malcolm G. Davis软件开发习惯中一个细微更改都可能会对软件质量产生巨大改进。将单元测试合并到开发过程中,然后从长远角度来看它可以节省多少时间和精力。本文通过使用代码样本说明了单元测试的种种好处,特别是使用 Ant 和 JUnit 带来的各种方便。 测试是大型开发过程中的基本原则之一。在任何职业中,验证都是一个重要部分。医生要通过验血来确诊。波音公司在研制 777 的过程中对飞机的每个组件都进行了精心测试。为什么软件开发就应该例外呢?以前,由于在应用程序中将 GUI 和商业逻辑紧密联系在一起,这就限制了创建自动测试的能力。当我们学会通过抽象层

2、将商业逻辑从界面中分离出来时,各个单独代码模块的自动测试就替代了通过 GUI 进行的手工测试。现在,集成开发环境 (IDE) 能在您输入代码的同时显示错误,对于在类中快速查找方法具有智能探测功能,可以利用语法结构生成彩色代码,而且具有许多其它功能。因此,在编译更改过的代码之前,您已经全盘考虑了将构建的类,但您是否考虑过这样的修改会破坏某些功能呢?每个开发者都碰到过更改“臭虫”。代码修改过程可能会引入“臭虫”,而如果通过用户界面手工测试代码的话,在编译完成之前是不会发现它的。然后,您就要花费几天的时间追踪由更改所引起的错误。最近在我做的一个项目中,当我把后端数据库由 Informix 更改到 O

3、racle 时就遇到了这种情况。大部分更改都十分顺利,但由于数据库层或使用数据库层的系统缺少单元测试,从而导致将大量时间花费在尝试解决更改“臭虫”上。我花了两天的时间查到别人代码中的一个数据库语法更改。(当然,那个人仍是我的朋友。)尽管测试有许多好处,但一般的程序员对测试都不太感兴趣,开始时我也没有。您听到过多少次“它编译了,所以它一定能用”这种言论?但“我思,故我在”这种原则并不适用于高质量软件。要鼓励程序员测试他们的代码,过程必须简单无痛。本文从某人学习用 Java 语言编程时所写的一个简单的类开始。然后,我会告诉您我是如何为这个类编写单元测试,以及在编写完它以后又是如何将单元测试添加到构

4、建过程中的。最后,我们将看到将“臭虫”引入代码时发生的情况。从一个典型类开始第一个典型的 Java 程序一般都包含一个打印 Hello World 的 main()。在清单 1 中,我创建了一个 HelloWorld 对象的实例并调用 sayHello() 方法,该方法会打印这句习惯说法。清单 1. 我的第一个 Java 应用程序 Hello world /* HelloWorld.java* My first java program*/class HelloWorld /* Print Hello World*/void sayHello() System.out.println(Hell

5、o World);/* Test*/public static void main( String args ) HelloWorld world = new HelloWorld();world.sayHello();main() 方法是我的测试。哦噢!我将代码、文档、测试和样本代码包含在了一个模块中。保佑 Java!但随着程序越变越大,这种开发方法很快就开始显现出了缺陷: 混乱 类接口越大,main() 就越大。类可能仅仅因为正常的测试而变得非常庞大。代码膨胀由于加入了测试,所以产品代码比所需要的要大。但我不想交付测试,而只想交付产品。测试不可靠既然 main() 是代码的一部分,main

6、() 就对其他开发者通过类接口无法访问的私有成员和方法享有访问权。出于这个原因,这种测试方法很容易出错。很难自动测试要进行自动测试,我仍然必须创建另一程序来将参数传递给 main()。类开发对我来 说,类开发是从编写 main() 方法开始的。我在编写 main() 的时候就定义类和类的用法,然后实现接口。它的一些明显的缺陷也开始显现出来。一个缺陷是我传递给 main() 来执行测试的参数个数。其次,main() 本身在进行调用子方法、设置代码等操作时变得很混乱。有时 main() 会比类实现的其余部分还要大。更简单的过程我原来的做法有一些很明显的缺陷。因此,让我们看看有什么别的方法可以使问题

7、简化。我仍然通过接口设计代码并给出应用示例,正如原来的 main() 一样。不同的是我将代码放到了另一个单独的类中,而这个类恰好是我的“单元测试”。这种技术有以下几点好处: 设计类的一种机制因为是通过接口进行开发,所以不太可能利用类的内部功能。但因为我是目标类的开发者,我有到其内部工作的“窗口”,所以测试并不是个真正的黑箱。仅凭这一点就足够推断出需要开发者本人在编写目标类的同时负责测试的开发,而不是由其他任何人代劳。类用法的示例通过将示例从实现中分离出来,开发者可以更快地提高速度,而且再不用在源代码上纠缠不清。这种分离还有助于防止开发者利用类的内部功能,因为这些功能将来可能已经不存在了。没有类

8、混乱的 main()我不再受到 main() 的限制了。以前我得将多个参数传递给 main() 来测试不同的配置。现在我可以创建许多单独的测试类,每一个都维护各自的设置代码。接下来我们将这个单独的单元测试对象放入构建过程中。这样,我们就可以提供自动确认过程的方法。 确保所做的任何更改都不会对其他人产生不利影响。我们在进行源码控制之前就可以测试代码,而无需等待汇编测试或在夜晚进行的构建测试。这有助于尽早捕捉到“臭虫”,从而降低产生高质量代码的成本。通过提供增量测试过程,我们提供了更好的实现过程。如同 IDE 帮助我们在输入时捕捉到语法或编译“臭虫”一样,增量单元测试也帮助我们在构建时捕捉到代码更

9、改“臭虫”。使用 JUnit 自动化单元测试要使测试自动化,您需要一个测试框架。您可以自己开发或购买,也可以使用某些开放源代码工具,例如 JUnit。我选择 JUnit 出于以下几个原因: 不需要编写自己的框架。它是开放源代码,因此不需要购买框架。开放源代码社区中的其他开发者会使用它,因此可以找到许多示例。它可以让我将测试代码与产品代码分开。它易于集成到我的构建过程中。测试布局图 1 显示了使用样本 TestSuite 的 JUnit TestSuite 布局。每个测试都由若干单独的测试案例构成。每个测试案例都是一个单独的类,它扩展了 TestClass 类并包含了我的测试代码,即那些曾在 m

10、ain() 中出现的代码。在该例中,我向 TestSuite 添加了两个测试:一个是 SkeletonTest,我将它用作所有新类和 HelloWorld 类的起点。图 1. TestSuite 布局测试类 HelloWorldTest.java按照约定,测试类的名 称中包含我所测试的类的名称,但将 Test 附加到结尾。在本例中,我们的测试类是 HelloWorldTest.java。我复制了 SkeletonTest 中的代码,并添加了 testSayHello() 来测试 sayHello()。请注意 HelloWorldTest 扩展了 TestCase。JUnit 框架提供了 ass

11、ert 和 assertEquals 方法,我们可以使用这些方法来进行验证。HelloWorldTest.java 显示在清单 2 中。 清单 2. HelloWorldTest.javapackage pany;import pany.HelloWorld;import junit.framework.TestCase;import junit.framework.AssertionFailedError; /* JUnit 3.2 testcases for HelloWorld*/public class HelloWorldTest extends TestCase public He

12、lloWorldTest(String name) super(name);public static void main(String args) junit.textui.TestRunner.run(HelloWorldTest.class);public void testSayHello() HelloWorld world = new HelloWorld();assert( world!=null );assertEquals(Hello World, world.sayHello() );testSayHello() 看上去和 HelloWorld.java 中原来的 main

13、 方法类似,但有一个主要的不同之处。它不是执行 System.out.println 并显示结果,而是添加了一个 assertEquals() 方法。如果两个值不同,assertEquals 将打印出两个输入的值。您可能已经注意到这个方法不起作用!HelloWorld 中的 sayHello() 方法不返回字符串。如果我先写过测试,就会捕捉到这一点。我将 Hello World 字符串与输出流联结起来。这样,按照清单 3 中显示的那样重写了 HelloWorld,去掉 main(),并更改了 sayHello() 的返回类型。清单 3. Hello world 测试案例。package pan

14、y;public class HelloWorld public String sayHello() return Hello World;如果我保留了 main() 并修改了联系,代码看上去如下: public static void main( String args ) HelloWorld world = new HelloWorld();System.out.println(world.sayHello();新的 main() 与我测试程序中的 testSayHello() 非常相似。是的,它看上去不象是一个现实世界中的问题(这是人为示例的问题),但它说明了问题。在单独的应用程序中编

15、写 main() 可以改进您的设计,同时帮助您设计测试。现在我们已经创建了一个测试类,让我们使用 Ant 来将它集成到构建中。使用 Ant 将测试集成到构建中Jakarta Project 将 Ant 工具说成“不带 make 缺点的 make”。Ant 正在成为开放源代码世界中实际上的标准。原因很简单:Ant 是使用 Java 语言编写的,这种语言可以让构建过程在多种平台上使用。这种特性简化了在不同 OS 平台之间的程序员的合作,而合作是开放源代码社区的一种需要。您可以在自己选择的平台上进行开发和构建。Ant 的特性包括: 类可扩展性Java 类可用于扩展构建特性,而不必使用基于 shell 的命令。开放源代码因为 Ant 是开放源代码,因此类扩展示例很充足。我发现通过示例来学习非常棒。XML 可配置Ant 不仅是基于 Java 的,它还使用 XML 文件配置构建过程。假设构建实际上是分层的,那么使用 XML 描述 make 过程就是其逻辑层。另外,如果您了解 XML,要学习如何配置构建就更简单一些。图 2 简要介绍了一个配置文件。配置文件由目标树构成。每个目标都包

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 生活休闲 > 社会民生

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号