《Java8實戰》-讀書筆記第一章(01)

雷俠發表於2019-02-28

Blog地址:www.codedream.xin

最近一直想寫點什麼東西,卻不知該怎麼寫,所以就寫寫關於看《Java8實戰》的筆記吧。

第一章內容較多,因此打算分幾篇文章來寫。

為什麼要關心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");
    }
}
複製程式碼

都要這寫麼多程式碼,不使用快捷鍵或者語法模板去生成,真的很浪費時間。(初學者請忽略)

所以,為了解決這個問題,Java8中推出了核心新特性之一:Lambda(匿名函式)
Lambda表示式,是一個很不錯很實用的一個新特性,如果你使用了這個新特性,也許會愛不釋手。

舉個例子,比如我們對蘋果進行按照重量進行排序,也許我們會這樣寫:

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的方法嗎?使用這個方法就可以判斷哪些是隱藏檔案啦。
是的,如下所示:

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,中文意思是謂詞:

什麼是謂詞?

前面的程式碼傳遞了方法Apple::isGreenApple(它接受引數Apple並返回一個boolean)給filterApples。後者希望接受一個Predicate引數。謂詞(predicate)在數學上常常用來代表一個類似函式的東西,它接受一個引數值,並返回true或false。

當然,Java8中已經有了一個Predicate介面,因此,我們也不需要去定義一個這樣的介面啦。

使用自定義的Predicate介面中的方法,你建立了一個方法引用,你無須去關注test方法是如何實現的,你只要知道你引用的某個方法即可。

讀書筆記系列的文章,因為我的文筆不是很好,一些觀點可能描述的不是那麼的好,文章中有誤的地方還請各位讀者指正,非常感謝。

相關文章