提取方法(Extract Method)是最常用的重构手法之一。当看到一个方法过长或者方法很难让人理解其意图的时候,这时候就可以用提取方法这种重构手法。
下面是我学习这个重构手法的笔记:
提取方法看起来好像仅仅是将被提取方法中的一段代码,放到目标方法中。其实,当方法足够复杂的时候,提取方法也会变得复杂。当然,如果提取方法这种重构手法无法进行时,就可能需要选择其他重构手法来进行了。
为了更方便的看出重构的效果,重构前的代码为:
public void printOwing() { double outstanding = 0.0; Enumeration e = _orders.elements(); System.out.println("****************************"); System.out.println("******* Customer Owes ******"); System.out.println("****************************"); while (e.hasMoreElements()) { Order v = (Order) e.nextElement(); outstanding += v.getAmount(); } // print result System.out.println("name: " + _name); System.out.println("amount: " + outstanding); }
提取方法分为三种常见的情况,这三种情况分三个步骤进行:
第1步 无局部变量
public void printOwing() { double outstanding = 0.0; Enumeration e = _orders.elements(); printBanner(); while (e.hasMoreElements()) { Order v = (Order) e.nextElement(); outstanding += v.getAmount(); } System.out.println("name: " + _name); System.out.println("amount: " + outstanding); } public void printBanner() { System.out.println("****************************"); System.out.println("******* Customer Owes ******"); System.out.println("****************************"); }
第2步 有局部变量,但目标方法仅读取它
public void printOwing() { double outstanding = 0.0; Enumeration e = _orders.elements(); printBanner(); // calculate outstanding while (e.hasMoreElements()) { Order v = (Order) e.nextElement(); outstanding += v.getAmount(); } printDetails(outstanding); } public void printBanner() { System.out.println("****************************"); System.out.println("******* Customer Owes ******"); System.out.println("****************************"); } public void printDetails(double outstanding) { System.out.println("name: " + _name); System.out.println("amount: " + outstanding); }
第3步 有局部变量,并且目标方法有对其进行赋值
public void printOwing() { printBanner(); double outstanding = getOutstanding(); printDetails(outstanding); } public void printBanner() { System.out.println("****************************"); System.out.println("******* Customer Owes ******"); System.out.println("****************************"); } public void printDetails(double outstanding) { System.out.println("name: " + _name); System.out.println("amount: " + outstanding); } public double getOutstanding() { double outstanding = 0.0; Enumeration e = _orders.elements(); while (e.hasMoreElements()) { Order v = (Order) e.nextElement(); outstanding += v.getAmount(); } return outstanding; }
总结一下,选择提炼方法这种重构手法有三种好处:
1.每个方法足够细粒度,那么就很方便进行代码复用
2.高层次的方法阅读的时候就像注释一样清晰明了
3.如果方法都是细粒度,方法的复写也会变得更容易一些