前言
上一章咱們了解了以下內(nèi)容:
① 為什么使用lambda表達式
② lambda表達式的語法
③ 函數(shù)式接口
這一章咱們繼續(xù)了解lambda表達式。
1.4 方法引用
咱們先看一下方法引用和非方法引用的區(qū)別:
//非方法引用
button.setOnAction(event->Sysout.out.println(event));
//方法引用
button.setOnAction(Sysout.out:println);
正如上面所示:
“ :: ”操作符將方法名和對象或類分隔開來奕短。以下是三種主要情況:
① 對象::實例方法
② 類::靜態(tài)方法
③ 類::實例方法
在①和②中方法引用等同于方法參數(shù)的lambda表達式。如之前所述沉御,Sysout.out::println等同于Sysout.out.println(x)窃爷。相似地粤剧,Math::pow等同于(x,y)->Math.pow(x,y)痘括。
在③中长窄,第一個參數(shù)會成為執(zhí)行方法的對象。例如:String::compareToIgnoreCase等同于(x,y)->x.compareIgnoreCase(y)纲菌。
注意:如果有多個同名的重載方法挠日,那么編譯器會從上下文中找到最匹配的一個方法。例如翰舌,如果有兩個版本的Math.max方法嚣潜,一個接收整型作為參數(shù),而另一個接收Double類型作為參數(shù),究竟會選擇哪個方法战虏,取決于Math.max被轉(zhuǎn)換為函數(shù)式接口的方法參數(shù)。同lambda表達式類似蝙寨,方法引用也不會獨立存在计技,它們經(jīng)常被用于轉(zhuǎn)換為函數(shù)式接口的實例喜德。
你還可以捕獲方法引用中的this參數(shù)。例如酸役,this::equals等同于x->this.equals(x)。當然你也可以使用super對象驾胆,例如涣澡,super::實例方法
以下用來說明該機制的示例:
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();
}
}
當線程啟動時,會調(diào)用它的Runnable方法丧诺,然后執(zhí)行super::greet并調(diào)用父類中的greet方法入桂。
注意:在匿名類中,你可以調(diào)用一個閉合類或父類的員工方法驳阎,例如閉合類.this::方法或閉合類.super::方法抗愁。
1.5 構(gòu)造器引用
構(gòu)造器引用同方法引用類似,不同的是構(gòu)造器引用中方法名是new呵晚。例如:Button:new表示Button類的構(gòu)造器引用蜘腌。
你可以使用數(shù)組類型來編寫構(gòu)造器引用。例如:int[]::new是一個含有一個參數(shù)的構(gòu)造器引用饵隙,這個參數(shù)就是數(shù)組的長度撮珠。它等同于lambda表達式x->new int[x]。數(shù)組構(gòu)造器可以用來繞過java中的一個限制金矛。在Java中芯急,無法構(gòu)造一個泛型類型T的數(shù)組。表達式new T[n]是錯誤的驶俊,因為它會被擦除為new Object[n]娶耍。這對于編寫API的開發(fā)人員來說是一個問題。例如饼酿,我們希望構(gòu)造一組按鈕榕酒,Stream接口中有一個返回Object數(shù)組的toArray方法:
Object[] buttons=stream.toArray();
但是這并不能讓我們滿意。用戶希望一組按鈕對象而并不是一組Object對象故俐,Stream API通過構(gòu)造器引用解決了這個問題奈应。它允許將Button[]::new 傳給toArray方法:
Button[] buttons=stream.toArray(Button[]::new);
toArray方法會調(diào)用該構(gòu)造器來獲得一個正確類型的數(shù)組。然后它會填充并返回該數(shù)組购披。