第三步:建立测试
选择File > New > Other 弹出新建对话框选择JUnit > TestCase(图4) >Next进入下一个对话框,在Test Calss中填入要测试的类(我这里是Person),Test Case会自动填写为PersonTest(图5),按Finish完成对测试用例的建立。代码如清单二。
图4:新建测试用例
图5:新建测试用例
清单二:
import junit.framework.TestCase;
public class PersonTest extends TestCase {
public PersonTest(String name) {
super(name);
}
public void testGetVehicle() {
assertEquals("Vroom", controlCar.getVehicle());
}
public static void main(String[] args) {
junit.textui.TestRunner.run(new PersonTest("testGetVehicle"));
}
protected void setUp() throws Exception {
super.setUp();
controlCar = new Person("Car");
controlTruck = new Person("Truck");
}
private Person controlCar;
private Person controlTruck;
}
第五步:对Person类进行测试
总算写好了这些类,让我来测试一番吧!选择菜单Run > Run…弹出Run对话框(图6),选择JUint单击New,在Test Class中填写测试类名(我这里为PersonTest),随后单击Run,将会有JUnit的图形界面出现在左边栏中,如果没出现请单击原package explorer栏中右下角的JUnit标签(图7)。其中绿色代表测试成功,红色则为失败。
图6:运行测试
图7:Juint的GUI
第六步:开始重构
总算轮到重构了,建立一个良好的测试也未尝不是对重构起到方便之处。好吧!从那里开始呢?先看看那些代码有什么坏味道?
很明显可以看到Car类和Truck类里的方法可以提取一个抽象基类。在解决这个坏味道之前,首先要对这两个类运用手法一进行值域的封装。在Car类中用鼠标反选engineStarted变量,单击右键选择Refactor > Encapsulate Field…(图8)会弹出封装值域的对话框,不用填写任何东西,按Ok完成值域的封装。运行测试看有无错误(测试非常重要,要做一步,测试一次)。当然你也可以按preview按钮查看它做了那些更改,如果有你不需要修改的地方大可去掉。同样完成对Truck类的值域封装。
图8:Refactor 菜单
接下来新建一个Vehicle类做为Car和Truck的基类,无需写任何代码。在Car类中继承Vehicle类(public class Car extends Vehicle)。好了现在可以把这些函数上移了(手法二)。在Car类中的选择go()方法右击,选择Refactor > Pull Up… 弹出上移对画框。设置如图9。你可以一路Next看看他是如何转换的,按Finish完成这次重构。运行测试通过。同样对Truck类进行这项重构。
图9:Pull Up 对话框
好了,让我们再来看看还能进行些什么重构?可以看到Truck类中的loadCargo()函数是卡车装载货物的方法。要是将来再有个货车类型,也需要此方法。所以应该提炼这个接口。有了这个Eclipse自动重构工具,提炼接口可以变得如此简单。选择Truck类的类名,右击选择Refactor > Extract Interface… 弹出Extract Interface对话框(图10),在对话框中的Interface name中输入接口名(我这里是CargoTransport),然后选择loadCargo()函数,可以选择Preview按钮看看它做了那些更改,单击ok完成提炼接口。运行单元测试,显示条为绿色。Ok成功。
图10:Extract Interface 对话框
第七步:最后的修改
最后我们来修改一下Person类,看看重构后修改了那些结构,如何降低了修改成本?
Person类代码修改如清单三,再看一下重构后的类图(类图2)。
清单三:
public class Person {
public Person(String arg){
try {
Class vehicleClass = Class.forName(arg);
vehicle = (Vehicle)vehicleClass.newInstance();
vehicle.startEngine();
} catch(Exception e) {
e.printStackTrace();
}
}
public String getVehicle(){
return vehicle.go();
}
private Vehicle vehicle;
}
类图2
可以看出重构后的类有着良好的持续发展的能力,当加入一个新的货车类时,可以不必修改Person类,当然重构后的代码也遵循了OCP的一个原则减少类之间的耦合,在抽象层上建立类之间的关联。这――就是这次重构带来的威力。
结束语:
Eclipse 所提供的重构工具使重构变得简单易行,重构可以提高你的编程速度,那么熟悉这些工具将更加有助于提高您的效率。敏捷开发方法采用迭代方式增加程序特性,因此需要依赖于重构技术来改变和扩展程序的设计。当然Eclipse所提供的重构工具不一定非要用在重构上,在你平时编码时一样可以派得上用场,不仅在进行一般的代码修改时提供节约时间的方法。还可以在不修改代码是一样使用(如值域的封装)。如果您花些时间熟悉这些工具,那么当出现可以利用它们的情况时,您就能意识到所花费的时间是值得的。
注:
本文的例子可能举的有些牵强,但这并不影响使用Eclipse实战重构。
参考资料
资源
l 本文重构后的原代码: http://sxhv998.y365.com/java/Source/traffic.rar
书籍
l 《重构-改善既有代码的设计》 作者 Martin Fowler
Web 站点
l http://www.refactoring.com/ Martin Fowler 的个人网站-是 Web 上的重构技术中心。
l http://www.junit.org/ 有关用于 JUnit 进行单元测试的更多信息。
l http://www.eclipse.org/ 有关Eclipse的更多信息(Eclipse2.1有十几种重构手法,最新的3.0版有多达三十几种重构手法)