重构、分支语句、虚函数、抽象函数与多态--《重构:改善既有代码设计》之读书心得
最近在读 Marting Fowler 所著的 Refactoring Improving the Desgin of Existing Code 一书。身体力行地完成了" a First Exmaple"--从一个现有代码的重构过程。
我是用VS.NET 2005 beta2 来做的。虽然这个版本已经实现了不少重构的工具,诸如 Rename、Encapsulate field、Extract Method、Extract Interface、Change signature 等。但是仍然有一些重要的、有用的重构,如:Extract superclass、Move Member(from one class to another)和 Push Up Member没有实现。所以我在重构的过程中有时需要手工改代码。
这个First Exmaple例子的最初原型是非常简单的,它是一个影碟出租商店用来计算并打印顾客账单的程序。它会显示一个客户租借了哪些影碟和多长时间,根据租借时间的长短计算费用。它还会识别:常规类、儿童类、新发布类三种不同的影碟类型。在计算租借费用的同时它还会计算客户的积分,如果租借的是新发布类的影碟会有额外的积分。
下面是这个程序中的几个关键类:
![](/Images/OutliningIndicators/None.gif)
![](/Images/OutliningIndicators/ContractedBlock.gif)
![](/Images/dot.gif)
![](/Images/OutliningIndicators/None.gif)
![](/Images/OutliningIndicators/ContractedBlock.gif)
![](/Images/dot.gif)
![](/Images/OutliningIndicators/None.gif)
![](/Images/OutliningIndicators/ContractedBlock.gif)
![](/Images/dot.gif)
重构针对这三个类展开,先后采用了Extract Method、Move Method、Removing Temps、Replace Type Code with State/Strategy等方法,最后是Replace Conditional with Polymorphism。这个过程中产生了新类Price(abstract)以及它的三个子类RegularPrice、ChildrensPrice、RegularPrice。
在最后的重构过程中,我先把Price类的getCharge方法改为虚拟(virtual)的,然后依次从swich语句中抽取子类的实现。
抽取前的Price类:
![](/Images/OutliningIndicators/None.gif)
![](/Images/OutliningIndicators/ContractedBlock.gif)
![](/Images/dot.gif)
抽取前的RegularPrice类:
![](/Images/OutliningIndicators/None.gif)
![](/Images/OutliningIndicators/ContractedBlock.gif)
![](/Images/dot.gif)
抽取后的Price和RegularPrice:
![](/Images/OutliningIndicators/None.gif)
![](/Images/OutliningIndicators/ContractedBlock.gif)
![](/Images/dot.gif)
![](/Images/OutliningIndicators/None.gif)
![](/Images/OutliningIndicators/None.gif)
![](/Images/OutliningIndicators/ContractedBlock.gif)
![](/Images/dot.gif)
继续抽取直至所有实现都转移到子类中, 这时候Price类变成:
![](/Images/OutliningIndicators/None.gif)
![](/Images/OutliningIndicators/ContractedBlock.gif)
![](/Images/dot.gif)
实际上Price类的getPrice方法已经没有任何计算了,这时候也没有子类尚未实现getCharge方法,所以把这个类改为抽象的就顺其自然了。改完后Price变成:
![](/Images/OutliningIndicators/None.gif)
![](/Images/OutliningIndicators/ContractedBlock.gif)
![](/Images/dot.gif)
从这个过程中你可以看到getPrice是如何从一个普通的方法变成virtual、最后变成abstract的。从一定程度上来说abstract方法就是virtual方法的一个特殊形式,只不过这个修饰符约定了所有继承具体类都实现了这个方法,因而这个方法中不需要再有任何实现了。
有时候花一点时间干点体力活是有好处的!