国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Java > 正文

Java函數式編程(三):列表的轉化

2019-11-26 15:24:39
字體:
來源:轉載
供稿:網友

列表的轉化

將集合轉化成一個新的集合就和遍歷它一樣簡單。假設我們要將列表中的名字轉化成全大寫的。我們看下都有哪些實現方式。

Java中的字符串是不可變的,所以它沒法改變。我們可以生成新的字符串,用來替換列表中原有的元素。然而這樣做的話,原來列表就沒了;還有一個問題,原來的列表可能也是不可變的,比如Arrays.asList()生成的,所以修改原來的列表這招不行。還有一個缺點就是這樣做很難并行操作。

生成一個新的全大寫的列表是個不錯的選擇。

乍聽起來這個建議弱爆了;性能是我們都很關注的一個問題。令人吃驚的是,函數式編程通常要比命令式的性能要高,我們在153頁的性能問題中會講到。

我們先開始用這個集合生成一個大寫字母的新集合吧。

復制代碼 代碼如下:

final List<String> uppercaseNames = new ArrayList<String>();
for(String name : friends) {
uppercaseNames.add(name.toUpperCase());
}

在命令式的代碼中,我們先創建一個空列表,然后把大寫的名字填充進去,在遍歷原來列表的過程中,每次插入一個。為了改進成函數式的版本,我們第一步可以考慮采用19頁遍歷列表中提到的那個內部迭代器forEach來替換一下for循環,正如下例所示的那樣。
復制代碼 代碼如下:

final List<String> uppercaseNames = new ArrayList<String>();
friends.forEach(name -> uppercaseNames.add(name.toUpperCase()));
System.out.println(uppercaseNames);

我們用了內部迭代器,但還得新建一個列表,然后再把元素插入到里面。我們還可以進一步改進。

使用lambda表達式

一個新引入的Stream接口里面,有個map方法,它可以幫助我們遠離可變性,并使代碼看起來更簡潔。Steam有點像集合的迭代器,同時它還提供了流函數(fluent functions)的功能。使用這個接口的方法,我們可以把一系列調用給組合起來,使代碼讀起來就像描述問題的順序一樣,可讀性更強。

Steam的map方法可以用來將輸入序列轉化成一個輸出的序列――這和我們要做的工作非常匹配。

復制代碼 代碼如下:

friends.stream()
.map(name -> name.toUpperCase())
.forEach(name -> System.out.print(name + " "));
System.out.println();

JDK8中的所有集合都支持這個stream方法,它把集合封裝成一個Steam實例。map方法對Stream中的每個元素都調用了指定的lambda表達式或者代碼塊。map方法跟forEach方法很不一樣, forEach只是簡單的對集合中的元素執行了一下指定的函數。而map方法把lambda表達式的運行結果收齊起來,返回一個結果集。最后我們用forEach方法打印了所有的元素。

新集合中的名字全都是大寫的了:

復制代碼 代碼如下:

BRIAN NATE NEAL RAJU SARA SCOTT

map方法很適合把一個輸入集合轉化成一個新的輸出集合。這個方法確保了輸入輸出序列的元素的數量是相同的。然而輸入元素和輸出元素的類型可以是不一樣的。在這個例子中,我們輸入和輸出的都是字符串的集合。我們可以傳給map方法一段代碼,讓它返回比如說名字中包含字符的個數。這樣的話,輸入的還是字符串的序列,而輸出的卻是數字序列了,就像下面這樣。

復制代碼 代碼如下:

friends.stream()
.map(name -> name.length())
.forEach(count -> System.out.print(count + " "));

結果是每個名字中字母的個數:
復制代碼 代碼如下:

5 4 4 4 4 5

使用了lambda表達式的之后版本,避免了顯式的修改操作;這樣的代碼非常簡潔。這樣寫不再需要初始化空的集合以及那個垃圾變量了;這個變量乖乖的躲到了底層實現里面了。

使用方法引用

我們還可以使用方法引用讓它變得更簡潔一些。在需要傳入函數式接口的實現的地方,Java編譯器可以接受lambda表達式或者是方法引用。有了這個特性,用String::toUpperCase就可以替換掉name -> name.toUpperCase()了,就像這樣:

復制代碼 代碼如下:

friends.stream()
.map(String::toUpperCase)
.forEach(name -> System.out.println(name));

當參數傳入到這個生成的方法――函數式接口的抽象方法的實現――里面的時候,Java會去調用這個String參數的toUpperCase方法。這個參數引用在這里就隱藏起來了。像前面這種簡單的場景,我們可以用方法引用來替換掉lambda表達式;更多的內容看一下26頁的什么時候應該使用方法引用。

復制代碼 代碼如下:

小伙伴發問了:
什么時候應該使用方法引用?

當使用Java編程的時候,通常我們用lambda表達式的時候要比方法引用多得多。但這并不意味著方法引用不重要或者沒啥用處。當lambda表達式非常簡短的時候,它是一個很好的替代方案,它直接調用了實例方法或者靜態方法。也就是說,如果lambda表達式只是傳遞了一下參數給方法調用的話,我們應該改用方法引用。
像這樣的lambda表達式,有點像Tom Smykowski在電影上班一條蟲中講的那樣,它的工作就是"從客戶那把需求拿給軟件工程師"。因為這個,我把這種重構成方法引用的模式叫做上班一條蟲模式。
除了簡潔外,使用方法引用,方法名字本身的含義和作用可以更好的體現出來。
使用方法引用背后,編譯器起到了很關鍵的作用。方法引用的目標對象和參數都會從這個生成的方法里傳進來的參數那推導出來。這才使得你可以使用方法引用寫出比使用lambda表達式更簡潔的代碼。不過,如果參數在傳遞給方法之前或者調用結果在返回之后要被修改的話,這種便利的寫法我們就用不了了。

在前面這個例子中,方法引用是引用了一個實例方法。方法引用還可以引用一個靜態方法以及接受傳參的方法。后面我們會看到這樣的例子。

lambda表達式能幫助我們遍歷集合,并且進行集合的轉化。就像下面我們即將看到的,它還能幫助我們快速的從集合中選取一個元素。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 邹平县| 象山县| 武乡县| 前郭尔| 随州市| 河津市| 昔阳县| 拜城县| 平昌县| 永丰县| 花莲市| 濉溪县| 乌鲁木齐县| 玉环县| 浪卡子县| 玉田县| 晋宁县| 海原县| 三门县| 永康市| 大石桥市| 浦城县| 南昌县| 象山县| 余庆县| 望都县| 聂拉木县| 明水县| 永和县| 武穴市| 陇川县| 彰化市| 滨海县| 洛阳市| 西林县| 明溪县| 深水埗区| 遂昌县| 特克斯县| 理塘县| 嘉定区|