1. 引言
Java8中最受廣大開發(fā)中喜歡的變化之一是因?yàn)橐肓?lambda 表達(dá)式寄摆,因?yàn)檫@些表達(dá)式允許我們放棄匿名類谅辣,從而大大減少了樣板代碼,并提高了可讀性婶恼。
方法引用是lambda表達(dá)式的一種特殊類型桑阶。它們通常通過引用現(xiàn)有方法來創(chuàng)建簡單的lambda表達(dá)式。
方法引用包括以下四種類型:
- 靜態(tài)方法
- 特定對(duì)象的實(shí)例方法
- 特定類型的任意對(duì)象的實(shí)例方法
- 構(gòu)造方法
在本篇文章中勾邦,我們將探討Java中的方法引用蚣录。
2. 引用靜態(tài)方法
We'll begin with a very simple example, capitalizing and printing a list of Strings:
我們從一個(gè)非常簡單的示例開始,字符串轉(zhuǎn)成大寫并打泳炱:
List<String> messages = Arrays.asList("hello", "baeldung", "readers!");
我們可以通過簡單的lambda表達(dá)式直接調(diào)用 StringUtils.capitalize()
方法:
messages.forEach(word -> StringUtils.capitalize(word));
或者萎河,我們可以使用方法引用來簡單地引用 capitalize
靜態(tài)方法:
messages.forEach(StringUtils::capitalize);
注意,方法引用應(yīng)使用::
運(yùn)算符。
3. 引用特定對(duì)象的實(shí)例方法
為了演示這種類型的方法引用虐杯,我們新建以下這兩個(gè)類:
public class Bicycle {
private String brand;
private Integer frameSize;
// standard constructor, getters and setters
}
public class BicycleComparator implements Comparator {
@Override
public int compare(Bicycle a, Bicycle b) {
return a.getFrameSize().compareTo(b.getFrameSize());
}
}
創(chuàng)建一個(gè) BicycleComparator 對(duì)象來比較自行車尺寸:
BicycleComparator bikeFrameSizeComparator = new BicycleComparator();
我們可以使用lambda表達(dá)式按尺寸大小對(duì)自行車進(jìn)行排序玛歌,但需要指定兩個(gè)自行車實(shí)例進(jìn)行比較:
createBicyclesList().stream()
.sorted((a, b) -> bikeFrameSizeComparator.compare(a, b));
我們可以使用方法引用讓編譯器把句柄參數(shù)傳遞給我們:
createBicyclesList().stream()
.sorted(bikeFrameSizeComparator::compare);
4. 引用特定類型任意對(duì)象的實(shí)例方法
這種類型的方法引用與前面的示例類似,但不必創(chuàng)建自定義對(duì)象來執(zhí)行比較擎椰。
讓我們創(chuàng)建一個(gè)要排序的Integer 整數(shù)列表:
List<Integer> numbers = Arrays.asList(5, 3, 50, 24, 40, 2, 9, 18);
如果我們使用經(jīng)典的 lambda 表達(dá)式支子,這兩個(gè)參數(shù)都需要顯式傳遞,而使用方法引用則要簡單得多:
numbers.stream()
.sorted((a, b) -> a.compareTo(b));
numbers.stream()
.sorted(Integer::compareTo);
盡管它仍然是一行代碼达舒,但是方法引用更容易閱讀和理解值朋。
5. 引用構(gòu)造函數(shù)
我們可以像在第一個(gè)例子中引用靜態(tài)方法一樣引用構(gòu)造函數(shù)。唯一區(qū)別是需要使用new關(guān)鍵字巩搏。
現(xiàn)在我們用不同品牌的String列表創(chuàng)建一個(gè)Bicycle數(shù)組:
List<String> bikeBrands = Arrays.asList("Giant", "Scott", "Trek", "GT");
首先昨登,我們將向Bicycle類添加一個(gè)新的構(gòu)造函數(shù):
public Bicycle(String brand) {
this.brand = brand;
this.frameSize = 0;
}
接下來,我們將使用方法引用中的新構(gòu)造函數(shù)贯底,并從原始的String列表中生成一個(gè)Bicycle數(shù)組:
bikeBrands.stream()
.map(Bicycle::new)
.toArray(Bicycle[]::new);
注意如何使用方法引用調(diào)用Bicycle和Array構(gòu)造函數(shù)丰辣,從而使代碼看起來更加簡潔明了。
6. 其他示例和限制
目前為止丈甸,方法引用是一個(gè)使代碼非常清晰和易讀的好方法糯俗。但是尿褪,我們不能用它們來代替各種lambda表達(dá)式睦擂,因?yàn)樗鼈冇幸恍┚窒扌浴?/p>
它們的主要局限性是由于它們最大的優(yōu)點(diǎn):前一個(gè)表達(dá)式的輸出需要與引用的方法聲明的輸入?yún)?shù)匹配。
看看這個(gè)限制的例子:
createBicyclesList().forEach(b -> System.out.printf(
"Bike brand is '%s' and frame size is '%d'%n",
b.getBrand(),
b.getFrameSize()));
這個(gè)簡單的例子不能用方法引用來表示杖玲,因?yàn)樵谖覀兊睦又校?em>printf 方法需要3個(gè)參數(shù)顿仇,而使用createBicyclesList().forEach()只允許方法引用一個(gè)參數(shù)(Bicycle對(duì)象)。
最后摆马,我們研究下臼闻,如何創(chuàng)建一個(gè)可以從lambda表達(dá)式引用的no-operation函數(shù)。
在本例中囤采,我們希望使用lambda表達(dá)式而不使用其參數(shù)述呐。
首先,創(chuàng)建 doNothingAtAll 方法:
private static <T> void doNothingAtAll(Object... o) {
}
因?yàn)檫@是一個(gè)varargs方法蕉毯,它可執(zhí)行在任意 lambda 表達(dá)式中乓搬,而不管引用的對(duì)象或參數(shù)的數(shù)量。我們看看它的作用:
createBicyclesList()
.forEach((o) -> MethodReferenceExamples.doNothingAtAll(o));
7. 總結(jié)
在這篇文章中代虾,我們學(xué)習(xí)了Java中的方法引用进肯,以及如何使用它們來替換lambda表達(dá)式,從而提高了可讀性并闡明編程的意圖棉磨。
如果你覺得文章還不錯(cuò)江掩,記得關(guān)注公眾號(hào): 鍋外的大佬
鍋外的大佬博客