??在軟件工程中,一個眾所周知的問題就是,不管你做什么,用戶的需求肯定會變的。比如說,有一個應用程序是幫助農民了解自己的庫存的。這位農民可能想有一個查找庫存中所有的綠蘋果的功能。但是到了第二天,他可能告訴你,其實還想找出所有重量超過150g的蘋果。又過了兩天,農民有跑過來補充道,要是我可以找出所有既是綠色,重量也超過150g的蘋果,那就太好了。你要如何應對這樣不斷變化的需求呢?
??行為參數化就是可以幫助你處理處理頻繁變更的需求的一種軟件開發模式。總之,它意味著拿出一個代碼塊,把它準備好卻不去執行它。這個代碼塊以后可以被你的程序的其他部分調用,這就表示你可以推遲這個代碼塊的執行。例如,你可以將代碼塊作為參數傳給一個方法,在這個方法里面再去執行它。這樣,這個方法的行為就基于那個代碼塊被參數化了。
1.應對不斷變化的需求
??編寫能夠應對變化的需求的代碼并不容易。讓我們看看一個例子,我們會逐步地改進這個例子,以展示一些讓代碼更加靈活的最佳做法。就農場庫存程序而言,你必須實現一個從列表中篩選綠蘋果的功能。
(1) 初試牛刀:篩選綠蘋果
?? 第一個解決方案可能是下面這樣的:
private List<Apple> filterGreenApple(List<Apple> inventory) {
List<Apple> result = new ArrayList<>(); // 累積的蘋果列表
for (Apple apple : inventory) {
if ("green".equals(apple.getColor())) { //僅僅篩選出綠蘋果
result.add(apple);
}
}
return result;
}
??這里篩選條件是篩選綠蘋果。但是先在農名改主意了,他還要篩選紅蘋果,你該怎么做呢?簡單的方法就是復制這段代碼,將名字改為filterRedApple,然后更改if條件來匹配紅蘋果。然而,要是農民想要篩選多種顏色:淺綠色、暗紅色、黃色等等,這種方法就應付不了。一個良好的原則是在編寫類似的代碼之后,嘗試將其抽象化
(2) 再展身手:把顏色作為參數
??一種做法是給方法加一個參數,把顏色變成參數,這樣就能靈活的適應變化了:
private List<Apple> filterAppleByColor(List<Apple> inventory, String color) {
List<Apple> result = new ArrayList<>();
for (Apple apple : inventory) {
if (color.equals(apple.getColor())) {
result.add(apple);
}
}
return result;
}
??現在,只要像下面這樣調用方法,農民朋友就會滿意了:
List<Apple> greenApples = filterAppleByColor(inventory, "green");
List<Apple> redApples = filterAppleByColor(inventory, "red");
??太簡單了對吧?那我們在把例子弄得更加復雜一點兒。這位農民又跑過來和你說,要是能夠區分輕的蘋果和重的蘋果就太好了。重的蘋果一般是重量大于150g。于是你寫了下面的方法,用一個參數來應對不同的重量:
private List<Apple> filterAppleByWeight(List<Apple> inventory, int weight) {
List<Apple> result = new ArrayList<>();
for (Apple apple : inventory) {
if (apple.getWeight() > weight) {
result.add(apple);
}
}
return result;
}
??解決方法不錯,但是復制大部分的代碼來實現遍歷庫存,并對每一個蘋果應用篩選條件。這讓人有點失望,因為它打破了DRY(Don't Repeat Yourself,不要重復自己)的軟件工程原則。如果你想要改變篩選方式來提升性能?那就得修改所有方法的實現,而不是只改一個。從工程的工作量角度來看,這個代價太大了
2.行為參數化
??在上面已經看到了,你需要一種比添加很多參數更好的方法來應對變化的需求。讓我們后退一步來看看更高層次的抽象。一種可能的解決方案是對你的選擇標準進行建模。讓我們定義一個接口來對選擇標準進行建模:
public interface ApplePredicate {
boolean test(Apple apple);
}
??現在可以使用ApplePredicate的多個實例代表不同的選擇標準了,比如:
public class AooleHeavyWeightPericate implements ApplePredicate { //選出重的蘋果
@Override
public boolean test(Apple apple) {
return apple.getWeight() > 150;
}
}
public class AppleGreenColorPerdicate implements ApplePredicate {//選擇綠蘋果
@Override
public boolean test(Apple apple) {
return "green".equals(apple.getColor());
}
}
(1)根據抽象條件進行篩選
??使用ApplePredicate改過之后,filter方法看起來是這樣的:
private List<Apple> filterApple(List<Apple> inventory, ApplePredicate p){
List<Apple> result = new ArrayList<>();
for (Apple apple : inventory) {
if (p.test(apple)) {
result.add(apple);
}
}
return result;
}