提取方法(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.如果方法都是细粒度,方法的复写也会变得更容易一些