最近一直想寫點什麼東西,殊不知該怎麼寫,因此就寫寫關於看《Java8實戰》的筆記吧。
第一章內容較多,所以打算分幾篇文章來寫。程序員
自1996年JDK(1.0)發佈以來,Java已經受到了學生、項目經理和 程序員等一大批活躍的用戶的歡迎。這一語言極富活力,不斷被用在大大小小的項目裏。從Java1.1(1997)年一直到Java7(2011)年,Java經過增長新功能,不斷獲得良好的升級。Java8則是在2014年3月發佈的。那麼問題來了:爲何你應該關心Java8?編程
是的,Java一直被吐槽寫起來太囉嗦,沒有IDE的快捷鍵和自動生成,簡直就是在用生命寫代碼,由於太浪費時間。
例如,最簡單的HelloWorld:數據結構
public class HelloWorld { public static void main (String[] args) { System.out.println("HelloWorld"); } }
都要這寫麼多代碼,不使用快捷鍵或者語法模板去生成,真的很浪費時間。(初學者請忽略)app
因此,爲了解決這個問題,Java8中推出了核心新特性之一:Lambda(匿名函數)
Lambda表達式,是一個很不錯很實用的一個新特性,若是你使用了這個新特性,也許會愛不釋手。ide
舉個例子,好比咱們對蘋果進行按照重量進行排序,也許咱們會這樣寫:函數式編程
private static List<Apple> apples = Arrays.asList(new Apple(100, "red"), new Apple(101, "green"), new Apple(132, "green"), new Apple(90, "green"), new Apple(122, "red") );
Collections.sort(apples, new Comparator<Apple>() { public int compare(Apple o1, Apple o2) { return o1.getWeight() < o2.getWeight() ? -1 : ((o1.getWeight() == o2.getWeight()) ? 0 : 1); } });
在Java8裏,你能夠這樣寫,這樣寫看起來更接近問題的描述:函數
apples.sort(Comparator.comparing(Apple::getWeight));
是否是有點心動啊,原本須要五六行解決的排序的代碼,如今只要一行便可!趁熱打鐵,繼續吧。
Java8裏面將代碼傳遞給方法的功能(同時也可以放回代碼並將其包含在數據結構中),還讓咱們可以使用一整套技巧,一般稱爲函數式編程。優化
如今你須要篩選一個目錄中的全部隱藏文件,你會怎麼作?
大部分人立馬會想到,File類中不就是有一個isHidden的方法嗎?使用這個方法就能夠判斷哪些是隱藏文件啦。
是的,以下所示:code
File[] files = new File("D:\\.").listFiles(new FileFilter() { @Override public boolean accept(File pathname) { return pathname.isHidden(); } });
看起來很簡單,很明瞭嘛!那還能夠不能夠繼續優化簡短一下呢?答案是固然能夠的。
以下所示:排序
File[] files = new File("D:\\.").listFiles(File::isHidden);
太酷了,有了函數isHidden,所以只須要使用Java8的方法引用語法::(即「把這個方法做爲值」)將其傳給listFiles方法就能夠了。
來看看一個例子,看看它是如何幫你你寫程序的。依舊使用剛剛對蘋果排序的代碼。如今,要作的是篩選出全部的綠蘋果,也許你會這一個這樣的方法filterGreenApples:
public static List<Apple> filterGreenApples(List<Apple> apples) { List<Apple> result = new ArrayList<>(); for (Apple apple : apples) { if ("green".equals(apple.getColor())) { result.add(apple); } } return result; }
在Java8以前,基本上都是這樣寫的,看起來也沒什麼毛病。可是,如今又要篩選一下重量超過120克的蘋果。哦,一想很簡單嘛,把上面的代碼複製、粘貼改一下就好啦:
public static List<Apple> filterHeavyApples(List<Apple> apples) { List<Apple> result = new ArrayList<>(); for (Apple apple : apples) { if (apple.getWeight() > 120) { result.add(apple); } } return result; }
雖然簡單,可是仍是出現了一些重複地方,看起來不太好。這兩段代碼的差別只是條件不一樣,那麼只須要把接受重量的上下限做爲參數傳遞給filter就能夠了,使用Java8來優化一下這些代碼:
public static void main (String[] args) { // 篩選出綠色蘋果 List<Apple> greenApples = filterApples(apples, FilterApples::isGreenApple); System.out.println(greenApples); // 篩選重量大於120克的蘋果 List<Apple> heavyApples = filterApples(apples, FilterApples::isHeavyApple); System.out.println(heavyApples); } public static boolean isGreenApple(Apple apple) { return "green".equals(apple.getColor()); } public static boolean isHeavyApple(Apple apple) { return apple.getWeight() > 120; } public static List<Apple> filterApples(List<Apple> apples, Predicate<Apple> predicate) { List<Apple> result = new ArrayList<>(); for (Apple apple : apples) { if (predicate.test(apple)) { result.add(apple); } } return result; } interface Predicate<T> { /** * 根據給定的參數計算此謂詞 * * @param t * @return */ boolean test(T t); }
在這段代碼中,自定義了一個接口Predicate<T>,中文意思是謂詞:
前面的代碼傳遞了方法Apple::isGreenApple(它接受參數Apple並返回一個boolean)給filterApples。後者但願接受一個Predicate<Apple>參數。謂詞(predicate)在數學上經常用來表明一個相似函數的東西,它接受一個參數值,並返回true或false。
固然,Java8中已經有了一個Predicate<T>接口,所以,咱們也不須要去定義一個這樣的接口啦。
使用自定義的Predicate接口中的方法,你建立了一個方法引用,你無須去關注test方法是如何實現的,你只要知道你引用的某個方法便可。
讀書筆記系列的文章,由於個人文筆不是很好,一些觀點可能描述的不是那麼的好,文章中有誤的地方還請各位讀者指正,很是感謝。