Intellij IDEA 竟然把 Java8 的數(shù)據(jù)流問(wèn)題這么完美的解決掉了!
閱讀本文大概需要 3?分鐘。
來(lái)自:?sf.gg/a/1190000006033999
Java8在2014年三月發(fā)布了。我們打算將Pondus的所有生產(chǎn)服務(wù)器升級(jí)到這一新版本。從那時(shí)起,我們將大部分代碼庫(kù)遷移到lambda表達(dá)式、數(shù)據(jù)流和新的日期API上。我們也會(huì)使用Nashorn來(lái)把我們的應(yīng)用中運(yùn)行時(shí)發(fā)生改變的部分變成動(dòng)態(tài)腳本。
除了lambda,最實(shí)用的特性是新的數(shù)據(jù)流API。集合操作在任何我見(jiàn)過(guò)的代碼庫(kù)中都隨處可見(jiàn)。而且對(duì)于那些集合操作,數(shù)據(jù)流是提升代碼可讀性的好方法。
但是一件關(guān)于數(shù)據(jù)流的事情十分令我困擾:數(shù)據(jù)流只提供了幾個(gè)終止操作,例如reduce和findFirst屬于直接操作,其它的只能通過(guò)collect來(lái)訪問(wèn)。工具類(lèi)Collctors提供了一些便利的收集器,例如toList、toSet、joining和groupingBy。
例如,下面的代碼對(duì)一個(gè)字符串集合進(jìn)行過(guò)濾,并創(chuàng)建新的列表:
stringCollection
????.stream()
????.filter(e?->?e.startsWith("a"))
????.collect(Collectors.toList());
在遷移了300k行代碼到數(shù)據(jù)流之后,我可以說(shuō),toList、toSet、和groupingBy是你的項(xiàng)目中最常用的終止操作。所以我不能理解為什么不把這些方法直接集成到Stream接口上面,這樣你就可以直接編寫(xiě):
stringCollection
????.stream()
????.filter(e?->?e.startsWith("a"))
????.toList();
這在開(kāi)始看起來(lái)是個(gè)小缺陷,但是如果你需要一遍又一遍地編寫(xiě)這些代碼,它會(huì)非常煩人。
有toArray()方法但是沒(méi)有toList(),所以我真心希望一些便利的收集器可以在Java9中這樣添加到Stream接口中。是吧,Brian??_?
注:Stream.js是瀏覽器上的Java 8 數(shù)據(jù)流API的JavaScript接口,并解決了上述問(wèn)題。所有重要的終止操作都可以直接在流上訪問(wèn),十分方便。詳情請(qǐng)見(jiàn)API文檔。
無(wú)論如何,IntelliJ IDEA聲稱(chēng)它是最智能的Java IDE。所以讓我們看看如何使用IDEA來(lái)解決這一問(wèn)題。
使用 IntelliJ IDEA 來(lái)幫忙
IntelliJ IDEA自帶了一個(gè)便利的特性,叫做實(shí)時(shí)模板(Live Template)。如果你還不知道它是什么:實(shí)時(shí)模板是一些常用代碼段的快捷方式。例如,你鍵入sout并按下TAB鍵,IDEA就會(huì)插入代碼段System.out.println()。更多信息請(qǐng)見(jiàn)這里。
如何用實(shí)時(shí)模板來(lái)解決上述問(wèn)題?實(shí)際上我們只需要為所有普遍使用的默認(rèn)數(shù)據(jù)流收集器創(chuàng)建我們自己的實(shí)時(shí)模板。例如,我們可以創(chuàng)建.toList縮寫(xiě)的實(shí)時(shí)模板,來(lái)自動(dòng)插入適當(dāng)?shù)氖占?collect(Collectors.toList())。
下面是它在實(shí)際工作中的樣子:

構(gòu)建你自己的實(shí)時(shí)模板
讓我們看看如何自己構(gòu)建它。首先訪問(wèn)設(shè)置(Settings)并在左側(cè)的菜單中選擇實(shí)時(shí)模板。你也可以使用對(duì)話框左上角的便利的輸入過(guò)濾。

下面我們可以通過(guò)右側(cè)的+圖標(biāo)創(chuàng)建一個(gè)新的組,叫做Stream。接下來(lái)我們向組中添加所有數(shù)據(jù)流相關(guān)的實(shí)時(shí)模板。我經(jīng)常使用默認(rèn)的收集器toList、toSet、groupingBy 和 join,所以我為每個(gè)這些方法都創(chuàng)建了新的實(shí)時(shí)模板。
這一步非常重要。在添加新的實(shí)時(shí)模板之后,你需要在對(duì)話框底部指定合適的上下文。你需要選擇Java → Other,然后定義縮寫(xiě)、描述和實(shí)際的模板代碼。
//?Abbreviation:?.toList
.collect(Collectors.toList())
//?Abbreviation:?.toSet
.collect(Collectors.toSet())
//?Abbreviation:?.join
.collect(Collectors.joining("$END$"))
//?Abbreviation:?.groupBy
.collect(Collectors.groupingBy(e?->?$END$))
特殊的變量$END$指定在使用模板之后的光標(biāo)位置,所以你可以直接在這個(gè)位置上打字,例如,定義連接分隔符。
提示:你應(yīng)該開(kāi)啟"Add unambiguous imports on the fly"(自動(dòng)添加明確的導(dǎo)入)選項(xiàng),便于讓IDEA自動(dòng)添加
java.util.stream.Collectors的導(dǎo)入語(yǔ)句。選項(xiàng)在Editor → General → Auto Import中。
讓我們?cè)趯?shí)際工作中看看這兩個(gè)模板:
連接

分組

Intellij IDEA中的實(shí)時(shí)模板非常靈活且強(qiáng)大,你可以用它來(lái)極大提升代碼的生產(chǎn)力。你知道實(shí)時(shí)模板可以拯救生活的其它例子嗎?請(qǐng)讓我知道!
推薦閱讀:
微信掃描二維碼,關(guān)注我的公眾號(hào)
朕已閱?

