??我們都知道喉悴,人們不愿意使用那些很麻煩的功能或者概念拧烦。目前冻河,當(dāng)要把新的行為傳遞給filterApples方法的時(shí)候,你不得不聲明好幾個(gè)實(shí)現(xiàn)ApplePredicate接口的類(lèi)友绝,然后去實(shí)例化好幾個(gè)只會(huì)提到一次的ApplePredicate對(duì)象堤尾。下面的程序總結(jié)了你目前看到的一切。這個(gè)真是非常的啰嗦迁客,很費(fèi)時(shí)間郭宝!
public class AppleHeavyWeightPericate implements ApplePredicate { //選出重的蘋(píng)果
@Override
public boolean test(Apple apple) {
return apple.getWeight() > 150;
}
}
public class AppleGreenColorPerdicate implements ApplePredicate {//選擇綠蘋(píng)果
@Override
public boolean test(Apple apple) {
return "green".equals(apple.getColor());
}
}
public class Demo {
public static void main(String[] args) {
List<Apple> inventory = Arrays.asList(new Apple("green", 80), new Apple("green", 155), new Apple("red", 120));
List<Apple> greenApples = filterApple(inventory, new AppleGreenColorPerdicate());
List<Apple> heavyWeightApples= filterApple(inventory, new AppleHeavyWeightPericate());
}
private static 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;
}
}
??費(fèi)這么大勁兒真沒(méi)必要,能不能做的更好呢掷漱?Java有一個(gè)機(jī)制稱(chēng)為匿名類(lèi)粘室,它可以讓你同時(shí)聲明和實(shí)例化一個(gè)類(lèi)。它可以幫助你進(jìn)一步改善代碼卜范,讓它變得更加的簡(jiǎn)潔衔统。
1.匿名類(lèi)
??匿名類(lèi)和你熟悉的Java局部類(lèi)(塊中定義的類(lèi))差不多,但匿名類(lèi)沒(méi)有名字海雪。它允許你同時(shí)聲明并且實(shí)例化一個(gè)類(lèi)锦爵。換句話說(shuō),它允許你隨用隨建奥裸。
(1).使用匿名類(lèi)
??下面的代碼展示了如何通過(guò)創(chuàng)建一個(gè)匿名類(lèi)實(shí)現(xiàn)ApplePredicate的對(duì)象险掀,重寫(xiě)篩選的例子:
List<Apple> redApples = filterApple(inventory, new ApplePredicate() {
@Override
public boolean test(Apple apple) {
return "red".equals(apple.getColor());
}
});
??但匿名類(lèi)還是不夠好。它往往顯得非常的笨重湾宙,因?yàn)檎加昧撕芏嗫臻g樟氢。比如
List<Apple> redApples = filterApple(inventory, new ApplePredicate() {
@Override
public boolean test(Apple apple) {
return "red".equals(apple.getColor());
}
});
List<Apple> greenApples = filterApple(inventory, new ApplePredicate() {
@Override
public boolean test(Apple apple) {
return "green".equals(apple.getColor());
}
});
??上面兩段代碼中,除了人return不同之外侠鳄,其余的代碼都是相同的埠啃。
??整體來(lái)說(shuō),啰嗦就不好伟恶。它讓人不愿意使用語(yǔ)言的某種功能碴开,因?yàn)榫帉?xiě)和維護(hù)啰嗦的代碼需要很多時(shí)間,而且代碼也易讀博秫。好的代碼應(yīng)該是一目了然的叹螟。即使使用匿名類(lèi)處理在某種程度上蓋上了為一個(gè)接口聲明好幾個(gè)實(shí)體類(lèi)的啰嗦問(wèn)題鹃骂,但是它仍然不能令人滿意台盯。
(2)使用Lambda表達(dá)式
??上面的代碼在java8里可以使用Lambda表達(dá)式重寫(xiě)為下面的樣子
List<Apple> result = filterApple(inventory, (Apple apple) ->"red".equals(apple.getColor()));
?? 不得不承認(rèn)這代碼看上去比先前干凈很多罢绽。這個(gè)很好,因?yàn)樗雌饋?lái)更像問(wèn)題稱(chēng)述本身了静盅。我們現(xiàn)在已經(jīng)解決了啰嗦的問(wèn)題了良价。
(3)將List類(lèi)型抽象化
?? 在通向抽象的路上,我們還可以更進(jìn)一步蒿叠。目前明垢,filterApples方法還只適用于Apple。你還可以將List類(lèi)型抽象化市咽,從而超越你眼前要處理的問(wèn)題
public interface ApplePredicate<T> {
boolean test(T t);
}
public class Demo {
public static void main(String[] args) {
List<Apple> inventory = Arrays.asList(new Apple("green", 80), new Apple("green", 155), new Apple("red", 120));
List<Apple> result = filterApple(inventory, (Apple apple) ->"red".equals(apple.getColor()));
for(Apple apple : result){
System.out.println(apple);
}
}
private static <T>List<T> filterApple(List<T> inventory, ApplePredicate<T> p) {
List<T> result = new ArrayList<>();
for (T t : inventory) {
if (p.test(t)) {
result.add(t);
}
}
return result;
}
}
??現(xiàn)在你可以把filter方法用在了香蕉痊银、橘子、Integer或是String的列表上了施绎∷莞铮酷不酷?你現(xiàn)在在靈活性和間接性之間找到了最佳的平衡點(diǎn)谷醉,這在Java8之前是不可能的
3.真實(shí)的例子
?&empsp;你現(xiàn)在已經(jīng)看到了致稀,行為參數(shù)化是一個(gè)很有用的模式,它能夠輕松的適應(yīng)不斷變化的需求俱尼。這種模式可以把一個(gè)行為(一段代碼)封裝起來(lái)抖单,并通過(guò)傳遞和使用創(chuàng)建的行為將方法的行為參數(shù)化。這種做法類(lèi)似于策略設(shè)計(jì)模式遇八。
(1).使用Comparator來(lái)排序
??對(duì)集合進(jìn)行排序是一個(gè)常見(jiàn)的編程任務(wù)矛绘。比如,你的那位農(nóng)民朋友想要根據(jù)蘋(píng)果的重量對(duì)庫(kù)存的進(jìn)行排序刃永,或者他可能改變了主意货矮,希望你根據(jù)顏色對(duì)蘋(píng)果進(jìn)行排序,聽(tīng)起來(lái)怎么有點(diǎn)耳熟呢揽碘?是的次屠,你需要一種方法來(lái)表示和使用不同的排序行為,來(lái)輕松的適應(yīng)變化的需求雳刺。
??在Java8中劫灶,List自帶一個(gè)sort方法(你也可以使用Collections.sort)。sort的行為可以用Java.util.Comparator對(duì)象來(lái)進(jìn)行參數(shù)化掖桦,它的接口如下:
public interface Comparator<T> {
int compare(T o1, T o2);
}
??因此本昏,你可以隨時(shí)創(chuàng)建Comparator的實(shí)現(xiàn),用sort方法表現(xiàn)出不同的行為枪汪。比如涌穆,你可以使用匿名類(lèi)怔昨,按照重量升序?qū)?kù)存進(jìn)行排序。
inventory.sort(new Comparator<Apple>() {
@Override
public int compare(Apple o1, Apple o2) {
return o1.compareTo(o2);
}
});
??用Lambda表達(dá)式的話宿稀,看起來(lái)就是這樣:
inventory.sort((Apple o1, Apple o2) ->o1.compareTo(o2));
(2).使用Runnable執(zhí)行代碼塊
?? 線程就像是輕量級(jí)的進(jìn)程:它們自己執(zhí)行一個(gè)代碼塊趁舀。但是怎么才能告訴線程要執(zhí)行那塊代碼呢?多個(gè)線程可能會(huì)運(yùn)行不同的代碼祝沸。我們需要一種方式來(lái)代表稍后執(zhí)行的一段代碼矮烹。在Java里面,你可以使用Runnable接口表示一個(gè)要執(zhí)行的代碼塊罩锐。請(qǐng)注意奉狈,代碼不會(huì)返回任何結(jié)果(即void):
??你可以像下面這樣,使用這個(gè)接口創(chuàng)建執(zhí)行不同行為的線程:
Thread t = new Thread(new Runnable() {
@Override
public void run() {
System.out.println("hello world");
}
});
??使用Lambda表達(dá)式涩惑,看起來(lái)是這樣的:
Thread t = new Thread(()->System.out.println("hello world"));