上一篇文章我們了解了Java8新特性-接口默認(rèn)方法蜕衡,接下來(lái)我們聊一聊Java8新特性之Lambda表達(dá)式缚甩。
Lambda表達(dá)式(也稱為閉包),它允許我們將函數(shù)當(dāng)成參數(shù)傳遞給某個(gè)方法,或者把代碼本身當(dāng)作數(shù)據(jù)處理吴侦。很多語(yǔ)言(Groovy挖胃、Scala等)從設(shè)計(jì)之初就支持Lambda表達(dá)式杂靶。但是java中使用的是?匿名內(nèi)部類代替。
最后借助強(qiáng)大的社區(qū)力量酱鸭,找了一個(gè)折中的Lambda實(shí)現(xiàn)方案吗垮,可以實(shí)現(xiàn)簡(jiǎn)潔而緊湊的語(yǔ)言結(jié)構(gòu)。
1凛辣、匿名內(nèi)部類到Lambda的演化
匿名內(nèi)部類抱既,即一個(gè)沒有名字的,存在于一個(gè)類或方法內(nèi)部的類扁誓。當(dāng)我們需要用某個(gè)類且只需要用一次防泵,創(chuàng)建和使用和二為一時(shí),我們可以選擇匿名內(nèi)部類蝗敢,省掉我們定義類的步驟捷泞。
匿名內(nèi)部類會(huì)隱士的繼承一個(gè)類或?qū)崿F(xiàn)一個(gè)接口,或者說(shuō)匿名內(nèi)部類是一個(gè)繼承了該類或者實(shí)現(xiàn)了該接口的子類匿名對(duì)象寿谴。下面看一個(gè)匿名內(nèi)部類的例子:
測(cè)試類中調(diào)用方法
package com.lotbyte.main;
/*
? ? 定義和使用匿名內(nèi)部類
*/
public class NoNameClass {
? ? public static void main(String[] args) {
? ? ? ? Model m = new Model(){
? ? ? ? ? ? @Override
? ? ? ? ? ? public void func() {
? ? ? ? ? ? ? ? System.out.println("方法的實(shí)現(xiàn)");
? ? ? ? ? ? }
? ? ? ? };
? ? ? ? m.func();
? ? }
}
// 需要被實(shí)現(xiàn)的接口
interface Model{
? ? void func();
}
2锁右、Lambda快速使用
從某種意義上來(lái)說(shuō),Lambda表達(dá)式可以看作是匿名內(nèi)部類對(duì)象的簡(jiǎn)寫形式。最簡(jiǎn)單的Lambda表達(dá)式可以由 用逗號(hào)分隔的參數(shù)列表咏瑟、->符號(hào)和語(yǔ)句塊組成拂到。
注意:此時(shí)匿名內(nèi)部類只能實(shí)現(xiàn)接口,不能是繼承抽象類
例如將上面的例子做一個(gè)簡(jiǎn)化码泞,使用Lambda的形式如下:
public class NonameClassForLambda {
? ? public static void main(String[] args) {
? ? ? ? // Lambda方式簡(jiǎn)寫兄旬,方法實(shí)現(xiàn)可以很簡(jiǎn)單
? ? ? ? Model1 md = ()-> System.out.println("hello");
? ? ? ? md.func();
? ? ? ? // 也可以是比較復(fù)雜的操作
? ? ? ? md = () -> {
? ? ? ? ? ? for (int i = 1; i <=5; i++) {
? ? ? ? ? ? ? ? System.out.println(i);
? ? ? ? ? ? }
? ? ? ? };
? ? ? ? md.func();
? ? }
}
// 接口
interface Model1{
? ? void func();
}
以上是一個(gè)簡(jiǎn)單的Lambda的書寫形式,()中是形參列表余寥,沒有則為空括號(hào)领铐,?->為語(yǔ)法格式,之后則為方法的實(shí)現(xiàn)(一條語(yǔ)句可以直接書寫宋舷,當(dāng)有多條語(yǔ)句時(shí)绪撵,需要使用{}進(jìn)行包裹)嚣崭。從這可以看出在接口中必須只能存在一個(gè)抽象方法淫奔。
注意:Lambda中必須有個(gè)接口
3、Lambda的形式
使用Lambda時(shí)偏序,實(shí)現(xiàn)方法可以有參數(shù)续膳,也可以有返回值改艇,如果沒指定參數(shù)類型,則由編譯器自行推斷得出坟岔。
3.1谒兄、 無(wú)參帶返回值
生成[1,10]之間的任意整數(shù)
說(shuō)明:Lambda的改寫需要有對(duì)應(yīng)的抽象方法,當(dāng)沒有參數(shù)時(shí)需要使用()占位社付,當(dāng)表達(dá)式只有一行代碼時(shí)承疲,可以省略return和{}
interface Model2{
? ? int func();
}
Model2 md2 = () -> {return (int)(Math.random()*10+1)};
以上的Lambda等價(jià)于:
Model2md2=()->(int)(Math.random()*10+1);
3.2 、帶參帶返回值
返回一個(gè)對(duì)數(shù)字描述的字符串
interface Model3{
? ? String func(int a);
}
Model3 md3 = (int a) -> {
? ? return "This is a number " + a;
};
說(shuō)明:形參寫在()內(nèi)即可鸥咖,參數(shù)的類型可以省略燕鸽,此時(shí)將由編譯器自行推斷得出,同時(shí)還可以省略()
md3=a->"This is a number "+a;
省略了參數(shù)類型啼辣,小括號(hào)啊研,同時(shí)連帶實(shí)現(xiàn)體的括號(hào)和return都省了。
3.3 鸥拧、帶多個(gè)參數(shù)
根據(jù)輸入的運(yùn)算符計(jì)算兩個(gè)數(shù)的運(yùn)算党远,并返回結(jié)果:
interface Model4{
? ? String func(int a, int b, String oper);
}
Model4 md4 = (a, b, s) -> {
? ? ? String res = "";
? ? ? if("+".equals(s)){
? ? ? ? ? ? res = ( a+b ) + "";
? ? ? }else if("-".equals(s)){
? ? ? ? ? ? res = ( a-b ) + "";
? ? ? }else if("*".equals(s)){
? ? ? ? ? ? res = ( a*b ) + "";
? ? ? }else if("/".equals(s)){
? ? ? ? ? ? res = ( a/b ) + ""; // 暫不考慮除0的情況
? ? ? }else{
? ? ? ? ? ? res =? "操作有失誤";
? ? ? }
? ? ? return res;
};
System.out.println(md4.func(1,1,"+"));
以上例子為多個(gè)參數(shù)的Lambda表達(dá)式,其中省略掉了每一個(gè)參數(shù)的類型富弦,編譯器自動(dòng)推斷沟娱。多條語(yǔ)句時(shí)實(shí)現(xiàn)體的{}不能省。
4腕柜、Lambda作為參數(shù)
在jdk8之前济似,接口可以作為方法參數(shù)傳入矫废,執(zhí)行時(shí)必須提供接口實(shí)現(xiàn)類的實(shí)例。從java8開始砰蠢,Lambda可以作為接口方法實(shí)現(xiàn)蓖扑,當(dāng)作參數(shù)傳入,無(wú)論從形式上還是實(shí)際上都省去了對(duì)象的創(chuàng)建娩脾。使代碼更加的緊湊簡(jiǎn)單高效赵誓。
使用Lambda表達(dá)式需要有以下幾步:
? 1、定義接口柿赊,抽象方法的模板;
? 2幻枉、在某方法中需要接口作為參數(shù)碰声;
? 3、調(diào)用方法時(shí)需要將抽象方法實(shí)現(xiàn)(此時(shí)我們使用Lambda表達(dá)式)并傳入即可熬甫。
4.1胰挑、定義接口
在接口中,必須有且僅有一個(gè)抽象方法椿肩,以確定Lambda模板
// 無(wú)參無(wú)返回值的方法
interface LambdaInterface1{
? ? void printString();
}
// 帶參無(wú)返回值的方法
interface? LambdaInterface2{
? ? void printString(String str);
}
4.2瞻颂、定義方法接收參數(shù)
在某方法中需要使用接口作為參數(shù)
// 無(wú)參
public static void testLambda(LambdaInterface1 lam1){
? ? lam1.printString();
}
// 帶參
public static void testLambda2(String s,LambdaInterface2 lam2){
? ? lam2.printString(s);
}
4.3、Lambda實(shí)現(xiàn)
使用方法時(shí)需要用Lambda將抽象方法實(shí)現(xiàn)
使用方法時(shí)需要用Lambda將抽象方法實(shí)現(xiàn)
// 無(wú)參Lambda作為參數(shù)
testLambda(()->{
? ? System.out.println("可以簡(jiǎn)單郑象,可以復(fù)雜");
});
// 帶參Lambda作為參數(shù)
testLambdaParam("hello",(a)->{
? ? System.out.println(a);
});
通過以上三步贡这,能夠完整地展示Lambda從和演變而來(lái)。此后在使用時(shí)厂榛,jdk中已經(jīng)提供很多場(chǎng)景了盖矫,即前兩部已經(jīng)完成,我們更多的是實(shí)現(xiàn)第三步即可击奶。
5辈双、forEach展示Lambda
例如以ArrayList的遍歷為例子,分析Lambda的使用方式柜砾。
public static void main(String[] args) {
? ? List<String> strs = new ArrayList<String>(){
? ? ? ? {
? ? ? ? ? ? add("aaa");
? ? ? ? ? ? add("bbb");
? ? ? ? ? ? add("ccc");
? ? ? ? }
? ? };
? ? strs.forEach((str)-> System.out.println(str));
}
下面看看forEach的源碼湃望,定義中使用了接口Consumer作為參數(shù),并調(diào)用了其方法:
Consumer中的抽象方法只有accept一個(gè)
通過在forEach方法中調(diào)用Consumer的accept方法痰驱,并將每一個(gè)元素作為參數(shù)傳入证芭,使得accept方法可以對(duì)每一個(gè)元素進(jìn)行操作,當(dāng)我們使用Lambda實(shí)現(xiàn)accept時(shí)就變成了我們自己對(duì)每一個(gè)元素的處理了萄唇。我們只負(fù)責(zé)處理即可檩帐。
6、Lambda中使用變量
? 在Lambda中可以定義自己的局部變量另萤,也可以使用外層方法的局部變量湃密,還可以使用屬性诅挑。這一點(diǎn)也不難理解,既然是一個(gè)方法的實(shí)現(xiàn)泛源,只寫了一個(gè)代碼塊拔妥,那么使用本身所屬方法的局部變量和類的屬性也并不過分。
public static void main(String[] args) {
? ? List<String> strs = new ArrayList<String>(){
? ? ? ? {
? ? ? ? ? ? add("aaa");
? ? ? ? ? ? add("bbb");
? ? ? ? ? ? add("ccc");
? ? ? ? }
? ? };
? ? int j = 1;
? ? strs.forEach((str)->{
? ? ? ? int i = 0;
? ? ? ? System.out.println(str + "? " + i + "? " + j);
? ? });
}
注意:此時(shí)外部局部變量將自動(dòng)變?yōu)閒inal
7达箍、Lambda作為方法返回值
例子:返回判斷字符串是否為空
public class Demo004_2 {
? ? public static void main(String[] args) {
? ? ? ? System.out.println(testLambda().isEmpty("string"));
? ? }
? ? // 判斷字符串是否為空
? ? public static AssertEmpty testLambda(){
? ? ? ? return (n)-> null==n||n.trim().isEmpty(n);
? ? }
}
interface AssertEmpty{
? ? boolean isEmpty(String str);
}
今天關(guān)于Java8新特性-Lambda表達(dá)式就講到這里了没龙,接下來(lái)我會(huì)繼續(xù)講述Java8新特性之函數(shù)式接口。敬請(qǐng)繼續(xù)關(guān)注缎玫!歡迎留言評(píng)論硬纤。