1.1 為什么要使用lambda表達(dá)式
"lambda 表達(dá)式"是一段可以傳遞的代碼砸逊,因此它可以被執(zhí)行一次或多次璧南。下面讓我們來(lái)下面的代碼示例:
class Worker implements Runnable{
public void run(){
for(int i=0;i<1000;i++){
doWork();
}
}
...
}
然后直接啟動(dòng)一個(gè)線程
Worker worker=new Worker();
new Thread(worker).start();
你會(huì)看到,一段代碼會(huì)被傳遞給其他調(diào)用者痹兜,也許是一個(gè)線程又或者是一個(gè)按鈕穆咐,這段代碼會(huì)稍后被調(diào)用。
到現(xiàn)在為止字旭,在Java中傳遞一段代碼并不容易对湃,你不可能將代碼塊到處傳遞。由于Java是一個(gè)面向?qū)ο笳Z(yǔ)言遗淳,你不得不構(gòu)建一個(gè)屬于某個(gè)類的對(duì)象拍柒,由它的方法來(lái)包含所需的方法。
雖然我們已經(jīng)通過(guò)類屈暗、對(duì)象的方式在Java中實(shí)現(xiàn)相似的功能拆讯,但是這使用起來(lái)并不讓人輕松和愉快。
1.2 lambda表達(dá)式的語(yǔ)法
lambda表達(dá)式在Java中的語(yǔ)法 :
1养叛、參數(shù)
2种呐、箭頭(->)
3、表達(dá)式
如果負(fù)責(zé)計(jì)算的代碼無(wú)法用一個(gè)表達(dá)式表示弃甥,那么可以用編寫方法的方式來(lái)編寫:即用{}包裹代碼并明確使用return語(yǔ)句爽室,例如:
(String first,String second)->{
if(first.length<second.length) return -1;
else if(first.length>second.length) return 1;
else return 0;
}
如果lambda表達(dá)式?jīng)]有參數(shù),你仍可以提供一對(duì)小括號(hào)()淆攻,如果不含參數(shù)的方法那樣:
()->{ for(int i=0;i<1000;i++) doWork(); }
如果一個(gè)lambda表達(dá)式的參數(shù)類型是可以被推導(dǎo)的阔墩,那么就可以省略它們的類型,例如:
Comparator<String>com=(first,second)//同(String first,String second)一樣
->Integer.compare(first.length(),second.length());
這里編譯器會(huì)推導(dǎo)出first和second必須是字符串瓶珊,因?yàn)閘ambda表達(dá)式被賦給了一個(gè)字符串比較器啸箫。
如果某個(gè)方法只含有一個(gè)參數(shù),并且該參數(shù)的類型也可以被推導(dǎo)出來(lái)伞芹,你甚至可以省略小括號(hào):
EventHandler<ActionEvent>listener=event-> Sysout.out.println("Thanks for clicking);
注意:你可以像對(duì)待方法參數(shù)一樣向lambda表達(dá)式的參數(shù)添加注解或者final修飾符忘苛,如下。
(final String name)->...
(@NonNull String name)->...
永遠(yuǎn)不需要為一個(gè)lambda表達(dá)式執(zhí)行返回類型,它總是會(huì)從上下文中被推導(dǎo)出來(lái)柑土。例如蜀肘,表達(dá)式
(String frist,String second)->Integer.compare(first.length(),second.length());
可以被使用在期望結(jié)果類型為int的上下文中。
注意:在lambda表達(dá)式中稽屏,只在某些分支中返回值(其它分支沒(méi)有返回值)是不合法的扮宠。例如,(int x)->{ if(x>0) return1; }是不合法的狐榔。
1.3 函數(shù)式接口
定義:只包含一個(gè)抽象方法的接口坛增,被稱之為函數(shù)式接口。
注意薄腻;你可能奇怪為什么函數(shù)式接口必須只有一個(gè)抽象方法收捣。難道接口中的方法不都是抽象的嗎?事實(shí)上庵楷,接口會(huì)經(jīng)常重新聲明Object類方法罢艾,例如toString或clone。而這些方法聲明并不是抽象的尽纽。(Java API中的某些接口重新聲明Object類方法咐蚯,是為了關(guān)聯(lián)javadoc的注釋。具體例子可以參考Comparator API)弄贿。之后你將會(huì)看到Java 8中接口可以聲明非抽象方法春锋。
為了演示函數(shù)式接口,我們以Arrays.sort方法為例差凹,該方法的第二個(gè)參數(shù)期奔,需要一個(gè)Comparator(該接口只含有一個(gè)方法)的實(shí)例,接下來(lái)危尿,我們來(lái)編寫一個(gè)簡(jiǎn)單的lambda表達(dá)式:
Arrays.sort(words,(first,second)->Integer.compare(first.length(),second.length()));
你最好將lambda表達(dá)式想象成一個(gè)函數(shù)呐萌,而不是一個(gè)對(duì)象,并記住它可以被轉(zhuǎn)換成一個(gè)函數(shù)式接口谊娇。
注意:你甚至不能將一個(gè)lambda表達(dá)式賦值給一個(gè)Object類型的變量肺孤,因?yàn)镺bject不是一個(gè)函數(shù)式接口。
請(qǐng)記住邮绿,任何一個(gè)lambda表達(dá)式都可以等價(jià)轉(zhuǎn)換成現(xiàn)在所使用的API中對(duì)應(yīng)的函數(shù)式接口渠旁。
注意:你可以在任意函數(shù)接口上標(biāo)注@FunctionalInterface注解攀例,這樣的好處有2個(gè)船逮。首先編譯器會(huì)檢查標(biāo)注該注解的實(shí)體,檢查它是否只包含一個(gè)抽象方法的接口粤铭。另外挖胃,在javadoc頁(yè)面也會(huì)包含一條聲明,說(shuō)明這個(gè)接口是一個(gè)函數(shù)式接口。該注解不要求強(qiáng)制使用酱鸭,從概念上來(lái)講吗垮,所有只含有一個(gè)抽象方法的接口都是函數(shù)式接口,但是使用@FunctionalInterface注解會(huì)讓你代碼看上去更清楚凹髓。
最后烁登,當(dāng)一個(gè)lambda表達(dá)式被轉(zhuǎn)換為一個(gè)函數(shù)式接口的實(shí)例時(shí),請(qǐng)注意檢查期異常蔚舀。如果lambda表達(dá)式中可能會(huì)拋出一個(gè)檢查期異常饵沧,那么該異常需要在目標(biāo)接口的抽象方法中聲明。例如赌躺,以下表達(dá)式會(huì)產(chǎn)生一個(gè)錯(cuò)誤:
Runnable sleeper=()->{ Sysout.out.println("zzz"); Thread.sleep(1000);};
//錯(cuò)誤:Thread.sleep可以拋出一個(gè)檢查期的InterruptedException狼牺。
由于Runnable.run不能拋出任何異常,所以這個(gè)賦值是不合法的礼患,有兩種方式可以修正該問(wèn)題是钥。
① 是在lambda表達(dá)式中捕獲該異常;
② 將lambda表達(dá)式賦給一個(gè)其抽象方法可以拋出異常的接口缅叠。
例如:Callable接口的call方法可以拋出任何異常悄泥,因此,你可以將lambda表達(dá)式賦給Callable<Void>(如果你添加一條" return null " 語(yǔ)句)痪署。
好了码泞,lambda表達(dá)式的基本形式、基本概念狼犯,到這里就結(jié)束了余寥。
接下來(lái)咱們會(huì)繼續(xù)了解lambda表達(dá)式的以下內(nèi)容:
① 方法引用
② 構(gòu)造器引用
③ 變量作用域