JBuilder2005實現重構之類內部提煉
1、提煉方法
對于一個方法體龐大的方法,可以使用Extract Method將其一部分代碼提煉到一個小方法中,并在原處添加提煉方法的調用。如果方法中存在重復的代碼段,則這個代碼段很有必須提煉為一個單獨的方法,以便復用。
要將一塊代碼提煉為一個單獨的方法,只需要選中這塊代碼,并通過Ctrl+Shift+R->Extract Method調出提煉方法的重構對話框來進行操作。JBuilder會分析選中代碼塊中所引用的上下文變量,目標方法定義相應的參數,通過方法入參來傳遞這些變量。如下面的drive()方法在Horse.java中定義,其代碼如下:
代碼清單 3 未提煉方法前的drive()方法
我們希望將第4'6行代碼塊抽取到一個稱為printDetail()的方法中:選中第4'6行的代碼->Ctrl+Shift+R->Extract Method調出提煉方法對話框,如下圖所示:
在Method name中鍵入printDetail,按OK后,JBuilder將選中的代碼塊提煉到新的方法中,并在原方法處插入提煉方法的調用:
代碼清單 4 進行方法提煉后的代碼
printDetail()方法有一個入參用于傳遞代碼塊所引用的上下文變量,在第4行的調用處將變量通過printDetail()的amount入參傳入。
2、內聯方法
內聯方法是提煉方法的反操作,即用方法體替換方法調用,當方法是private且再沒有其他地方調用該方法時,JBuilder在內聯方法后,一并刪除這個被內聯的方法。如我們將光標放置代碼清單 4的第4行printDetail()處:Ctrl+Shift+R->Inline Method,JBuilder直接將printDetail()方法體內聯到第4行處,因為在Horse.java中沒有其他地方調用printDetail()所以,該函數內在聯后將從Horse.java中刪除。內聯后代碼如代碼清單 3所示。
如果系統中所有函數都似乎只是對另一個函數的簡單委托,形成過多的間接層,過多的委托將使程序的關系變得不易理解,此時可以使用內聯方法去除這些無用的委托方法。
3、提煉變量
我們曾經說過“不同的方法有重復的代碼塊時,須考慮將其抽取為一個獨立的方法,不同的類有重復的方法時,須考慮抽取到一個新的類中”,相似的,如果一個方法中多次重復出現同一個表達式,此時你可以考慮將這些相同的表達式提煉為一個局域變量,請看下面的代碼: 代碼清單 5 重復的表達式
在printAccount()方法中表達式amount*price重復出現3次,我們可以在函數開始處引入一個局域變量,用這個變量替換這個表達式:Ctrl+Shift+R->Introduce Variable調出引入變量的對話框,如下圖所示:
引入變量對話框
·Name:totalPrice 引入的變量名。
·Type:變量類型,不可變,JBuilder自動斷定。
·Expression:你所選中的計算表達式。
·Include all read expressions:如果此選項勾選,則JBuilder將在方法范圍內查找所有重復的表達式,一并替換。你沒有什么理由不選擇這個設置項。
按OK后完成重構,重構后代碼如下所示:
代碼清單 6 引入局域變量去除重復表達式
特別的,如果一個表達式過于復雜,應當按特定意義劃分成多個部分,每個部分抽取成一個獨立的局域變量,以提高程序的可讀性。如下面是計算n個圓錐的造價:
可以將其重構為以下的形式:
4、內聯變量
和內聯方法一樣,內聯變量是提煉變量的反操作。有兩種內聯變量的方法,如果你將光標放置在變量的定義處進行內聯變量重構,所有引用該變量的地方都被替換,變量定義語句被刪除;如果你將光標置于引用變量處進行內聯變量操作,則可以有兩種選擇:只對當前的位置進行變量內聯或內聯變量的所有引用處。 我們使用代碼清單 6演示第二種內聯變量的方法。將光標置于第5行代碼的totalPrice處:Ctrl+Shift+R->Inline Variable調出內聯變量的重構對話框,如下圖所示:
內聯變量對話框
·Variable:選擇內聯的變量。
·Expression:內聯的表達式。
·Replace all occurrences:內聯所有引用totalPrice變量的地方,否則只內聯當前引用處。
點擊OK完成變量的內聯操作,內聯后代碼如代碼清單 5所示。
對于一個方法體龐大的方法,可以使用Extract Method將其一部分代碼提煉到一個小方法中,并在原處添加提煉方法的調用。如果方法中存在重復的代碼段,則這個代碼段很有必須提煉為一個單獨的方法,以便復用。
要將一塊代碼提煉為一個單獨的方法,只需要選中這塊代碼,并通過Ctrl+Shift+R->Extract Method調出提煉方法的重構對話框來進行操作。JBuilder會分析選中代碼塊中所引用的上下文變量,目標方法定義相應的參數,通過方法入參來傳遞這些變量。如下面的drive()方法在Horse.java中定義,其代碼如下:
代碼清單 3 未提煉方法前的drive()方法
| 1. public void printQuotation(int amount) 2. { 3. printBanner(); 4. System.out.println("單價:"+price); 5. System.out.println("購買只數:"+amount); 6. System.out.println("總價:"+(price * amount)); 7. } |
我們希望將第4'6行代碼塊抽取到一個稱為printDetail()的方法中:選中第4'6行的代碼->Ctrl+Shift+R->Extract Method調出提煉方法對話框,如下圖所示:
![]() 圖 12 提煉方法對話框 |
在Method name中鍵入printDetail,按OK后,JBuilder將選中的代碼塊提煉到新的方法中,并在原方法處插入提煉方法的調用:
代碼清單 4 進行方法提煉后的代碼
| 1. public void printQuotation(int amount) 2. { 3. printBanner(); 4. printDetail(amount);//在源處插入對抽取出方法的引用 5. } 6. //抽取出來的方法 7. private void printDetail(int amount) { 8. System.out.println("單價:"+price); 9. System.out.println("購買只數:"+amount); 10. System.out.println("總價:"+(price * amount)); 11. } |
printDetail()方法有一個入參用于傳遞代碼塊所引用的上下文變量,在第4行的調用處將變量通過printDetail()的amount入參傳入。
2、內聯方法
內聯方法是提煉方法的反操作,即用方法體替換方法調用,當方法是private且再沒有其他地方調用該方法時,JBuilder在內聯方法后,一并刪除這個被內聯的方法。如我們將光標放置代碼清單 4的第4行printDetail()處:Ctrl+Shift+R->Inline Method,JBuilder直接將printDetail()方法體內聯到第4行處,因為在Horse.java中沒有其他地方調用printDetail()所以,該函數內在聯后將從Horse.java中刪除。內聯后代碼如代碼清單 3所示。
如果系統中所有函數都似乎只是對另一個函數的簡單委托,形成過多的間接層,過多的委托將使程序的關系變得不易理解,此時可以使用內聯方法去除這些無用的委托方法。
3、提煉變量
我們曾經說過“不同的方法有重復的代碼塊時,須考慮將其抽取為一個獨立的方法,不同的類有重復的方法時,須考慮抽取到一個新的類中”,相似的,如果一個方法中多次重復出現同一個表達式,此時你可以考慮將這些相同的表達式提煉為一個局域變量,請看下面的代碼: 代碼清單 5 重復的表達式
| 1. public void printAccountPrice(int amount) 2. { 3. if(amount <= 100) { 4. System.out.println("總價:" + (amount * price)) ; 5. } 6. else if(amount <= 500) { 7. System.out.println("總價(打9折):" + (amount * price) * 0.9) ; 8. } 9. else { 10. System.out.println("總價(打9折):" + (amount * price) * 0.9) ; 11. } 12. } |
在printAccount()方法中表達式amount*price重復出現3次,我們可以在函數開始處引入一個局域變量,用這個變量替換這個表達式:Ctrl+Shift+R->Introduce Variable調出引入變量的對話框,如下圖所示:
![]() 圖 13 |
引入變量對話框
·Name:totalPrice 引入的變量名。
·Type:變量類型,不可變,JBuilder自動斷定。
·Expression:你所選中的計算表達式。
·Include all read expressions:如果此選項勾選,則JBuilder將在方法范圍內查找所有重復的表達式,一并替換。你沒有什么理由不選擇這個設置項。
按OK后完成重構,重構后代碼如下所示:
代碼清單 6 引入局域變量去除重復表達式
| 1. public void printAccountPrice(int amount) 2. { 3. double totalPrice = (amount * price); 4. if(amount <= 100) { 5. System.out.println("總價:" + totalPrice) ; 6. } 7. else if(amount <= 500) { 8. System.out.println("總價(打9折):" + totalPrice * 0.9) ; 9. } 10. else { 11. System.out.println("總價(打9折):" + totalPrice * 0.9) ; 12. } 13. } |
特別的,如果一個表達式過于復雜,應當按特定意義劃分成多個部分,每個部分抽取成一個獨立的局域變量,以提高程序的可讀性。如下面是計算n個圓錐的造價:
| n*Math.PI*r*r*h/3*price |
可以將其重構為以下的形式:
| double s = Math.PI*r*r*h/3;//圓錐體積 totalPrice = n * s * price;//n個圓錐造價 |
4、內聯變量
和內聯方法一樣,內聯變量是提煉變量的反操作。有兩種內聯變量的方法,如果你將光標放置在變量的定義處進行內聯變量重構,所有引用該變量的地方都被替換,變量定義語句被刪除;如果你將光標置于引用變量處進行內聯變量操作,則可以有兩種選擇:只對當前的位置進行變量內聯或內聯變量的所有引用處。 我們使用代碼清單 6演示第二種內聯變量的方法。將光標置于第5行代碼的totalPrice處:Ctrl+Shift+R->Inline Variable調出內聯變量的重構對話框,如下圖所示:
![]() 圖 14 |
內聯變量對話框
·Variable:選擇內聯的變量。
·Expression:內聯的表達式。
·Replace all occurrences:內聯所有引用totalPrice變量的地方,否則只內聯當前引用處。
點擊OK完成變量的內聯操作,內聯后代碼如代碼清單 5所示。


