??在軟件工程中啊易,一個眾所周知的問題就是延欠,不管你做什么,用戶的需求肯定會變的铡原。比如說偷厦,有一個應(yīng)用程序是幫助農(nóng)民了解自己的庫存的。這位農(nóng)民可能想有一個查找?guī)齑嬷兴械木G蘋果的功能燕刻。但是到了第二天只泼,他可能告訴你,其實還想找出所有重量超過150g的蘋果卵洗。又過了兩天请唱,農(nóng)民有跑過來補充道,要是我可以找出所有既是綠色过蹂,重量也超過150g的蘋果十绑,那就太好了。你要如何應(yīng)對這樣不斷變化的需求呢酷勺?
??行為參數(shù)化就是可以幫助你處理處理頻繁變更的需求的一種軟件開發(fā)模式本橙。總之脆诉,它意味著拿出一個代碼塊甚亭,把它準(zhǔn)備好卻不去執(zhí)行它。這個代碼塊以后可以被你的程序的其他部分調(diào)用击胜,這就表示你可以推遲這個代碼塊的執(zhí)行亏狰。例如,你可以將代碼塊作為參數(shù)傳給一個方法偶摔,在這個方法里面再去執(zhí)行它骚揍。這樣,這個方法的行為就基于那個代碼塊被參數(shù)化了。
1.應(yīng)對不斷變化的需求
??編寫能夠應(yīng)對變化的需求的代碼并不容易信不。讓我們看看一個例子嘲叔,我們會逐步地改進這個例子,以展示一些讓代碼更加靈活的最佳做法抽活。就農(nóng)場庫存程序而言硫戈,你必須實現(xiàn)一個從列表中篩選綠蘋果的功能。
(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;
}
??這里篩選條件是篩選綠蘋果下硕。但是先在農(nóng)名改主意了丁逝,他還要篩選紅蘋果,你該怎么做呢梭姓?簡單的方法就是復(fù)制這段代碼霜幼,將名字改為filterRedApple,然后更改if條件來匹配紅蘋果誉尖。然而罪既,要是農(nóng)民想要篩選多種顏色:淺綠色、暗紅色铡恕、黃色等等琢感,這種方法就應(yīng)付不了。一個良好的原則是在編寫類似的代碼之后探熔,嘗試將其抽象化
(2) 再展身手:把顏色作為參數(shù)
??一種做法是給方法加一個參數(shù)驹针,把顏色變成參數(shù),這樣就能靈活的適應(yīng)變化了:
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;
}
??現(xiàn)在诀艰,只要像下面這樣調(diào)用方法柬甥,農(nóng)民朋友就會滿意了:
List<Apple> greenApples = filterAppleByColor(inventory, "green");
List<Apple> redApples = filterAppleByColor(inventory, "red");
??太簡單了對吧?那我們在把例子弄得更加復(fù)雜一點兒其垄。這位農(nóng)民又跑過來和你說苛蒲,要是能夠區(qū)分輕的蘋果和重的蘋果就太好了。重的蘋果一般是重量大于150g捉捅。于是你寫了下面的方法撤防,用一個參數(shù)來應(yīng)對不同的重量:
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;
}
??解決方法不錯,但是復(fù)制大部分的代碼來實現(xiàn)遍歷庫存棒口,并對每一個蘋果應(yīng)用篩選條件寄月。這讓人有點失望,因為它打破了DRY(Don't Repeat Yourself,不要重復(fù)自己)的軟件工程原則无牵。如果你想要改變篩選方式來提升性能漾肮?那就得修改所有方法的實現(xiàn),而不是只改一個茎毁。從工程的工作量角度來看克懊,這個代價太大了
2.行為參數(shù)化
??在上面已經(jīng)看到了忱辅,你需要一種比添加很多參數(shù)更好的方法來應(yīng)對變化的需求。讓我們后退一步來看看更高層次的抽象谭溉。一種可能的解決方案是對你的選擇標(biāo)準(zhǔn)進行建模墙懂。讓我們定義一個接口來對選擇標(biāo)準(zhǔn)進行建模:
public interface ApplePredicate {
boolean test(Apple apple);
}
??現(xiàn)在可以使用ApplePredicate的多個實例代表不同的選擇標(biāo)準(zhǔn)了,比如:
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)根據(jù)抽象條件進行篩選
??使用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;
}