學(xué)習(xí)札記-Java8系列-8-強(qiáng)大的Stream API
什么是Stream
Java8中有兩大最為重要的特性。
1)Lambda 表達(dá)式,已經(jīng)學(xué)習(xí)過了
2)Stream API (java.util.stream.*包下)
說到Stream便容易想到I/O Stream,而實(shí)際上我們這里講的Stream它是Java8中對數(shù)據(jù)處理的一種抽象描述议慰;
我們可以把它理解為數(shù)據(jù)的管道肴楷,我們可以通過這條管道提供給我們的API很方便的對里面的數(shù)據(jù)進(jìn)行復(fù)雜的操作!比如查找焦人、過濾和映射(類似于使用SQL);
更厲害的是可以使用Stream API 來并行執(zhí)行操作;
簡而言之重父,Stream API 提供了一種高效且易于使用的處理數(shù)據(jù)的方式花椭,解決了已有集合類庫操作上的弊端。
●注意:
1.請暫時(shí)忘記對傳統(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ù)類型提供不同方法實(shí)現(xiàn)奕污。那什么不把IntStream等設(shè)計(jì)成Stream的子接口?畢竟這接口中的方法名大部分是一樣的液走。答案是這些方法的名字雖然相同碳默,但是返回類型不同,如果設(shè)計(jì)成父子接口關(guān)系缘眶,這些方法將不能共存嘱根,因?yàn)镴ava不允許只有返回類型不同的方法重載。
傳統(tǒng)集合操作vs Stream API操作
傳統(tǒng)方式遍歷集合
幾乎所有的集合(如Collection 接口或Map 接口等)都支持直接或間接的遍歷操作巷懈。而當(dāng)我們需要對集合中的元
素進(jìn)行操作的時(shí)候该抒,除了必需的添加、刪除顶燕、獲取外凑保,最典型的就是集合遍歷。例如:
List<String> list = new ArrayList<>();
list.add("馬云");
list.add("馬化騰");
list.add("李彥宏");
list.add("雷軍");
list.add("劉強(qiáng)東");
for (String name : list) {
System.out.println(name);
}
這是一段非常簡單的集合遍歷操作:對集合中的每一個(gè)字符串都進(jìn)行打印輸出操作涌攻。
循環(huán)遍歷的弊端
Java 8的Lambda讓我們可以更加專注于做什么(What)欧引,而不是怎么做(How),這點(diǎn)此前已經(jīng)結(jié)合匿名內(nèi)部類進(jìn)行了對比說明】一眩現(xiàn)在芝此,我們仔細(xì)體會(huì)一下上例代碼,可以發(fā)現(xiàn):
for循環(huán)的語法就是“怎么做”
for循環(huán)的循環(huán)體才是“做什么”(這才是我們要完成的目標(biāo)!)
如果覺得上面那樣的for循環(huán)也無所謂的話婚苹,我們來完成幾個(gè)需求:
1)找出姓馬的岸更;
2)再從姓馬的中找到名字長度等于3的。
如何實(shí)現(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("劉強(qiáng)東");
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);
}
}
這段代碼中含有三個(gè)循環(huán)坐慰,每一個(gè)作用不同:
1)首先篩選所有姓張的人;
2)然后篩選名字有三個(gè)字的人用僧;
3)最后進(jìn)行對結(jié)果進(jìn)行打印輸出。
每當(dāng)我們需要對集合中的元素進(jìn)行操作的時(shí)候赞咙,總是需要進(jìn)行循環(huán)责循、循環(huán)、再循環(huán)攀操。這是理所當(dāng)然的么院仿?不是。循環(huán)是做事情的方式速和,而不是目的歹垫。另一方面,使用線性循環(huán)就意味著只能遍歷一次颠放。如果希望再次遍歷排惨,只能再使用另一個(gè)循環(huán)從頭開始。那Stream能給我們帶來怎樣更加優(yōu)雅的寫法呢碰凶?
Stream的優(yōu)雅寫法
下面來看一下如果使用Java 8的Stream API來實(shí)現(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("劉強(qiáng)東");
list.stream()
.filter(s -> s.startsWith("馬"))
.filter(s -> s.length() == 3)
.forEach(System.out::println);
}
使用Stream API將我們真正想要做的事情直接體現(xiàn)在代碼中暮芭。直接閱讀代碼的字面意思即可完美展示我們要做的事:
獲取流、過濾出姓馬的欲低、過濾出名字長度為3的辕宏、逐一打印。
代碼如此簡潔砾莱,直觀瑞筐,優(yōu)雅!