Java 函數式編程(三)流(Stream)

流使程序猿可以在抽象層上對集合進行操作。

從外部迭代到內部迭代

什么是外部迭代和內部迭代呢?

個人認為,外和內是相對集合代碼而言。

如果迭代的業務執行在應用代碼中,稱之為外部迭代。

反之,迭代的業務執行在集合代碼中,稱為內部迭代(函數式編程)。

語言描述可能有點抽象,下面看實例。

1. 外部迭代

調用itrator方法,產生一個新的Iterator對象,進而控制整個迭代過程。

for (Student student:list){
    if (student.getAge()>18){
        result++;
    }
}

我們都知道,for其實底層使用的迭代器:

Iterator<Student> iterator = list.iterator();
while (iterator.hasNext()){
    Student student = iterator.next();
    if (student.getAge()>18){
        result++;
    }
}

上面的迭代方法就是外部迭代。

外部迭代缺點:
  1. 很難抽象出復雜操作
  2. 本質上講是串行化操作。

2. 內部迭代

返回內部迭代中的響應接口:Stream

long count = list.stream().filter(student -> student.getAge() > 18).count();

整個過程被分解為:過濾和計數。

要注意:返回的Stream對象不是一個新集合,而是創建新集合的配方。

2.1 惰性求值和及早求值

像filter這樣值描述Stream,最終不產生新集合的方法叫做惰性求值
像count這樣最終會從Stream產生值的方法叫做及早求值

判斷一個操作是惰性操作還是及早求值,只需看它的返回值。如果返回值是Stream,那么是惰性求值;如果返回值是另一個值或者為空,那就是及早求值。這些操作的理想方式就是形成一個惰性求值的鏈,最后用一個及早求值的操作返回想要的結果。

整個過程跟建造者模式很像,使用一系列的操作后最后調用build方法才返回真正想要的對象。設計模式快速學習(四)建造者模式

那這個過程有什么好處呢:可以在集合類上級聯多種操作,但迭代只需要進行一次。

3. 常用操作

3.1 collect(toList()) 及早求值

collect(toList())方法由Stream里的值生成一個列表,是一個及早求值操作。

List<String> collect = Stream.of("a", "b", "c").collect(Collectors.toList());

Stream.of("a", "b", "c")首先由列表生成一個Stream對象,然后collect(Collectors.toList())生成List對象。

3.2 map

map可以將一種類型的值轉換成另一種類型。

List<String> streamMap = Stream.of("a", "b", "c").map(String -> String.toUpperCase()).collect(Collectors.toList());

map(String -> String.toUpperCase())將返回所有字母的大寫字母的Stream對象,collect(Collectors.toList())返回List。

3.3 filter過濾器

遍歷并檢查其中的元素時,可用filter

List<String> collect1 = Stream.of("a", "ab", "abc")
        .filter(value -> value.contains("b"))
        .collect(Collectors.toList());
3.4 flatMap

如果有一個包含了多個集合的對象希望得到所有數字的集合,我們可以用flatMap

List<Integer> collect2 = Stream.of(asList(1, 2), asList(3, 4))
        .flatMap(Collection::stream)
        .collect(Collectors.toList());

Stream.of(asList(1, 2), asList(3, 4))將每個集合轉換成Stream對象,然后.flatMap處理成新的Stream對象。

3.5 max和min

看名字就知道,最大值和最小值。

Student student1 = list.stream()
        .min(Comparator.comparing(student -> student.getAge()))
        .get();

java8提供了一個Comparator靜態方法,可以借助它實現一個方便的比較器。其中Comparator.comparing(student -> student.getAge()可以換成Comparator.comparing(Student::getAge)成為更純粹的lambda。max同理。

3.6 reduce

reduce操作可以實現從一組值中生成一個值,在上述例子中用到的count、min、max方法事實上都是reduce操作。

Integer reduce = Stream.of(1, 2, 3).reduce(0, (acc, element) -> acc + element);
System.out.println(reduce);

6

上面的例子使用reduce求和,0表示起點,acc表示累加器,保存著當前累加結果(每一步都將stream中的元素累加至acc),element是當前元素。

4. 操作整合

  1. collect(toList())方法由Stream里的值生成一個列表
  2. map可以將一種類型的值轉換成另一種類型。
  3. 遍歷并檢查其中的元素時,可用filter
  4. 如果有一個包含了多個集合的對象希望得到所有數字的集合,我們可以用flatMap
  5. max和min
  6. reduce(不常用)

5. 鏈式操作實戰

List<Student> students = new ArrayList<>();
students.add(new Student("Fant.J",18));
students.add(new Student("小明",19));
students.add(new Student("小王",20));
students.add(new Student("小李",22));
List<Class> classList = new ArrayList<>();
classList.add(new Class(students,"1601"));
classList.add(new Class(students,"1602"));
    static class Student{
        private String name;
        private Integer age;
        getter and setter ...and construct ....
    }

    static class Class{
        private List<Student> students;
        private String className;
        getter and setter ...and construct ....
    }

這是我們的數據和關系--班級和學生,現在我想要找名字以小開頭的學生,用stream鏈式操作:

List<String> list= students.stream()
                            .filter(student -> student.getAge() > 18)
                            .map(Student::getName)
                            .collect(Collectors.toList());
[小明, 小王, 小李]

這是一個簡單的students對象的Stream的鏈式操作實現,那如果我想要在許多個class中查找年齡大于18的對象呢?

6. 實戰提升

在許多個class中查找年齡大于18的名字并返回集合。

原始代碼:

        List<String> nameList = new ArrayList<>();
        for (Class c:classList){
            for (Student student:c.getStudents()){
                if (student.getAge()>18){
                    String name = student.getName();
                    nameList.add(name);
                }
            }
        }

        System.out.println(nameList);

鏈式流代碼:
如果讓你去寫,你可能會classList.stream().forEach(aClass -> aClass.getStudents().stream())....去實現?

我剛開始就是這樣無腦干的,后來我緩過神來,想起foreach是一個及早求值操作,而且返回值是void,這樣的開頭就注定了沒有結果,然后仔細想想,flatMap不是用來處理不是一個集合的流嗎,好了,就有了下面的代碼。

List<String> collect = classList.stream()
        .flatMap(aclass -> aclass.getStudents().stream())
        .filter(student -> student.getAge() > 18)
        .map(Student::getName)
        .collect(toList());

原始代碼和流鏈式調用相比,有以下缺點

  1. 代碼可讀性差,隱匿了真正的業務邏輯
  2. 需要設置無關變量來保存中間結果
  3. 效率低,每一步都及早求值生成新集合
  4. 難于并行化處理

7. 高階函數及注意事項

高階函數是指接受另外一個函數作為參數,或返回一個函數的函數。如果函數的函數里包含接口或返回一個接口,那么該函數就是高階函數。

Stream接口中幾乎所有的函數都是高階函數。比如:Comparing 接受一個函數作為參數,然后返回Comparator接口。

Student student = list.stream().max(Comparator.comparing(Student::getAge)).get();

public interface Comparator<T> {}

foreach方法也是高階函數:

void forEach(Consumer<? super T> action);

public interface Consumer<T> {}

除了以上還有一些類似功能的高階函數外,不建議將lambda表達式傳給Stream上的高階函數,因為大部分的高階函數這樣用會有一些副作用:給變量賦值。局部變量給成員變量賦值,導致很難察覺。

這里拿ActionEvent舉例子:

?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,488評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,034評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,327評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,554評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,337評論 6 404
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,883評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,975評論 3 439
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,114評論 0 286
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,625評論 1 332
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,555評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,737評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,244評論 5 355
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 43,973評論 3 345
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,362評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,615評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,343評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,699評論 2 370

推薦閱讀更多精彩內容