學(xué)習(xí)札記-Java8系列-8-強大的Stream API
什么是Stream
Java8中有兩大最為重要的特性。
1)Lambda 表達式,已經(jīng)學(xué)習(xí)過了
2)Stream API (java.util.stream.*包下)
說到Stream便容易想到I/O Stream,而實際上我們這里講的Stream它是Java8中對數(shù)據(jù)處理的一種抽象描述;
我們可以把它理解為數(shù)據(jù)的管道,我們可以通過這條管道提供給我們的API很方便的對里面的數(shù)據(jù)進行復(fù)雜的操作!比如查找、過濾和映射(類似于使用SQL);
更厲害的是可以使用Stream API 來并行執(zhí)行操作;
簡而言之,Stream API 提供了一種高效且易于使用的處理數(shù)據(jù)的方式,解決了已有集合類庫操作上的弊端。
●注意:
1.請暫時忘記對傳統(tǒng)IO流的固有印象!
2.Stream接口繼承關(guān)系

圖中4種stream接口繼承自BaseStream,其中IntStream, LongStream, DoubleStream對應(yīng)三種基本類型(int, long, double,注意不是包裝類型),Stream對應(yīng)所有剩余類型的stream視圖。為不同數(shù)據(jù)類型設(shè)置不同stream接口,可以提高性能,并針對不同數(shù)據(jù)類型提供不同方法實現(xiàn)。那什么不把IntStream等設(shè)計成Stream的子接口?畢竟這接口中的方法名大部分是一樣的。答案是這些方法的名字雖然相同,但是返回類型不同,如果設(shè)計成父子接口關(guān)系,這些方法將不能共存,因為Java不允許只有返回類型不同的方法重載。
傳統(tǒng)集合操作vs Stream API操作
傳統(tǒng)方式遍歷集合
幾乎所有的集合(如Collection 接口或Map 接口等)都支持直接或間接的遍歷操作。而當(dāng)我們需要對集合中的元
素進行操作的時候,除了必需的添加、刪除、獲取外,最典型的就是集合遍歷。例如:
List<String> list = new ArrayList<>();
list.add("馬云");
list.add("馬化騰");
list.add("李彥宏");
list.add("雷軍");
list.add("劉強東");
for (String name : list) {
System.out.println(name);
}
這是一段非常簡單的集合遍歷操作:對集合中的每一個字符串都進行打印輸出操作。
循環(huán)遍歷的弊端
Java 8的Lambda讓我們可以更加專注于做什么(What),而不是怎么做(How),這點此前已經(jīng)結(jié)合匿名內(nèi)部類進行了對比說明?,F(xiàn)在,我們仔細(xì)體會一下上例代碼,可以發(fā)現(xiàn):
for循環(huán)的語法就是“怎么做”
for循環(huán)的循環(huán)體才是“做什么”(這才是我們要完成的目標(biāo)!)
如果覺得上面那樣的for循環(huán)也無所謂的話,我們來完成幾個需求:
1)找出姓馬的;
2)再從姓馬的中找到名字長度等于3的。
如何實現(xiàn)?在Java 8之前的做法可能為:
@Test
public void testWithFor() throws Exception {
List<String> list = new ArrayList<>();
list.add("馬云");
list.add("馬化騰");
list.add("李彥宏");
list.add("雷軍");
list.add("劉強東");
List<String> maList = new ArrayList<>();
for (String name : list) {
if (name.startsWith("馬")) {
maList.add(name);
}
}
List<String> length3List = new ArrayList<>();
for (String name : maList) {
if (name.length() == 3) {
length3List.add(name);
}
}
for (String name : length3List) {
System.out.println(name);
}
}
這段代碼中含有三個循環(huán),每一個作用不同:
1)首先篩選所有姓張的人;
2)然后篩選名字有三個字的人;
3)最后進行對結(jié)果進行打印輸出。
每當(dāng)我們需要對集合中的元素進行操作的時候,總是需要進行循環(huán)、循環(huán)、再循環(huán)。這是理所當(dāng)然的么?不是。循環(huán)是做事情的方式,而不是目的。另一方面,使用線性循環(huán)就意味著只能遍歷一次。如果希望再次遍歷,只能再使用另一個循環(huán)從頭開始。那Stream能給我們帶來怎樣更加優(yōu)雅的寫法呢?
Stream的優(yōu)雅寫法
下面來看一下如果使用Java 8的Stream API來實現(xiàn)將有多么的優(yōu)雅:
@Test
public void testWithStream() throws Exception {
List<String> list = new ArrayList<>();
list.add("馬云");
list.add("馬化騰");
list.add("李彥宏");
list.add("雷軍");
list.add("劉強東");
list.stream()
.filter(s -> s.startsWith("馬"))
.filter(s -> s.length() == 3)
.forEach(System.out::println);
}
使用Stream API將我們真正想要做的事情直接體現(xiàn)在代碼中。直接閱讀代碼的字面意思即可完美展示我們要做的事:
獲取流、過濾出姓馬的、過濾出名字長度為3的、逐一打印。
代碼如此簡潔,直觀,優(yōu)雅!