前言
上一章咱們了解了以下內(nèi)容:
① 為什么使用lambda表達(dá)式
② lambda表達(dá)式的語法
③ 函數(shù)式接口
這一章咱們繼續(xù)了解lambda表達(dá)式。
1.4 方法引用
咱們先看一下方法引用和非方法引用的區(qū)別:
//非方法引用
button.setOnAction(event->Sysout.out.println(event));
//方法引用
button.setOnAction(Sysout.out:println);
正如上面所示:
“ :: ”操作符將方法名和對(duì)象或類分隔開來。以下是三種主要情況:
① 對(duì)象::實(shí)例方法
② 類::靜態(tài)方法
③ 類::實(shí)例方法
在①和②中方法引用等同于方法參數(shù)的lambda表達(dá)式。如之前所述,Sysout.out::println等同于Sysout.out.println(x)。相似地,Math::pow等同于(x,y)->Math.pow(x,y)。
在③中,第一個(gè)參數(shù)會(huì)成為執(zhí)行方法的對(duì)象。例如:String::compareToIgnoreCase等同于(x,y)->x.compareIgnoreCase(y)。
注意:如果有多個(gè)同名的重載方法,那么編譯器會(huì)從上下文中找到最匹配的一個(gè)方法。例如,如果有兩個(gè)版本的Math.max方法,一個(gè)接收整型作為參數(shù),而另一個(gè)接收Double類型作為參數(shù),究竟會(huì)選擇哪個(gè)方法,取決于Math.max被轉(zhuǎn)換為函數(shù)式接口的方法參數(shù)。同lambda表達(dá)式類似,方法引用也不會(huì)獨(dú)立存在,它們經(jīng)常被用于轉(zhuǎn)換為函數(shù)式接口的實(shí)例。
你還可以捕獲方法引用中的this參數(shù)。例如,this::equals等同于x->this.equals(x)。當(dāng)然你也可以使用super對(duì)象,例如,super::實(shí)例方法
以下用來說明該機(jī)制的示例:
class Greeter{
public void greet(){
Sysout.out.println("Hello World");
}
}
class ChildGreeter extends Greeter{
public void greet(){
Thread t=new Thread(super::greet);
t.start();
}
}
當(dāng)線程啟動(dòng)時(shí),會(huì)調(diào)用它的Runnable方法,然后執(zhí)行super::greet并調(diào)用父類中的greet方法。
注意:在匿名類中,你可以調(diào)用一個(gè)閉合類或父類的員工方法,例如閉合類.this::方法或閉合類.super::方法。
1.5 構(gòu)造器引用
構(gòu)造器引用同方法引用類似,不同的是構(gòu)造器引用中方法名是new。例如:Button:new表示Button類的構(gòu)造器引用。
你可以使用數(shù)組類型來編寫構(gòu)造器引用。例如:int[]::new是一個(gè)含有一個(gè)參數(shù)的構(gòu)造器引用,這個(gè)參數(shù)就是數(shù)組的長度。它等同于lambda表達(dá)式x->new int[x]。數(shù)組構(gòu)造器可以用來繞過java中的一個(gè)限制。在Java中,無法構(gòu)造一個(gè)泛型類型T的數(shù)組。表達(dá)式new T[n]是錯(cuò)誤的,因?yàn)樗鼤?huì)被擦除為new Object[n]。這對(duì)于編寫API的開發(fā)人員來說是一個(gè)問題。例如,我們希望構(gòu)造一組按鈕,Stream接口中有一個(gè)返回Object數(shù)組的toArray方法:
Object[] buttons=stream.toArray();
但是這并不能讓我們滿意。用戶希望一組按鈕對(duì)象而并不是一組Object對(duì)象,Stream API通過構(gòu)造器引用解決了這個(gè)問題。它允許將Button[]::new 傳給toArray方法:
Button[] buttons=stream.toArray(Button[]::new);
toArray方法會(huì)調(diào)用該構(gòu)造器來獲得一個(gè)正確類型的數(shù)組。然后它會(huì)填充并返回該數(shù)組。