更新時(shí)間:2021-06-28 16:35:31 來(lái)源:動(dòng)力節(jié)點(diǎn) 瀏覽1273次
JDK8已經(jīng)發(fā)布快4年的時(shí)間了,現(xiàn)在來(lái)談它的新特性顯得略微的有點(diǎn)“不合時(shí)宜”。盡管JDK8已不再“新”,但它的重要特性之一——Lambda表達(dá)式依然是不被大部分開(kāi)發(fā)者所熟練運(yùn)用,甚至不被開(kāi)發(fā)者所熟知。
國(guó)內(nèi)的開(kāi)發(fā)環(huán)境大家都知道,有各種的老項(xiàng)目,有各種各樣的發(fā)布風(fēng)險(xiǎn),讓公司以及項(xiàng)目組對(duì)新的技術(shù)往往望而卻步,有公司甚至?xí)r至今日還在使用JDK6來(lái)進(jìn)行項(xiàng)目開(kāi)發(fā),這導(dǎo)致了在很多技術(shù)的選擇上受到了很大限制,進(jìn)而不能跟隨時(shí)代的腳步使得項(xiàng)目甚至公司一步一步走向衰落。
本文簡(jiǎn)單認(rèn)識(shí)JDK8的重要新特性之一——Lambda表達(dá)式。在JDK8之前,Java是不支持函數(shù)式編程的,所謂的函數(shù)編程,即可理解是將一個(gè)函數(shù)(也稱(chēng)為“行為”)作為一個(gè)參數(shù)進(jìn)行傳遞。通常我們提及得更多的是面向?qū)ο缶幊蹋嫦驅(qū)ο缶幊淌菍?duì)數(shù)據(jù)的抽象(各種各樣的POJO類(lèi)),而函數(shù)式編程則是對(duì)行為的抽象(將行為作為一個(gè)參數(shù)進(jìn)行傳遞)。在JavaScript中這是很常見(jiàn)的一個(gè)語(yǔ)法特性,但在Java中將一個(gè)函數(shù)作為參數(shù)傳遞這卻行不通,好在JDK8的出現(xiàn)打破了Java的這一限制。
認(rèn)識(shí)Lambda表達(dá)式
首先來(lái)引入一個(gè)示例,不知給是否有在IDEA編寫(xiě)代碼的經(jīng)歷,如果在JDK8的環(huán)境下如下所示按照J(rèn)ava傳統(tǒng)的語(yǔ)法規(guī)則編寫(xiě)一個(gè)線(xiàn)程。
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("Hello World!");
}
});
IDEA會(huì)給出提示可以使用Lambda表達(dá)式替換。
使用Lambda表達(dá)式則只需要使用一句話(huà)就可代替上面使用匿名類(lèi)的方式
new Thread(() -> System.out.println("Hello World!"));
在這個(gè)例子中,傳統(tǒng)的語(yǔ)法規(guī)則,我們是將一個(gè)匿名內(nèi)部類(lèi)作為參數(shù)進(jìn)行傳遞,我們實(shí)現(xiàn)了Runnable接口,并將其作為參數(shù)傳遞給Thread類(lèi),這實(shí)際上我們傳遞的是一段代碼,也即我們將代碼作為了數(shù)據(jù)進(jìn)行傳遞,這就帶來(lái)許多不必要的“樣板代碼”。
Lambda表達(dá)式一共有三部分組成:
后面的示例中我們會(huì)詳解這個(gè)結(jié)構(gòu),包括有無(wú)參數(shù),有無(wú)返回值的問(wèn)題。那么這個(gè)看起來(lái)奇奇怪怪的不太像Java的語(yǔ)法規(guī)則,其本身含義到底什么呢?這也是開(kāi)始困擾我的問(wèn)題,什么時(shí)候在什么場(chǎng)景下可以使用Lambda表達(dá)式。
能夠接收Lambda表達(dá)式的參數(shù)類(lèi)型,是一個(gè)只包含一個(gè)方法的接口。只包含一個(gè)方法的接口稱(chēng)之為“函數(shù)接口”。
例如上面創(chuàng)建一個(gè)線(xiàn)程的示例,Runnable接口只包含一個(gè)方法,所以它被稱(chēng)為“函數(shù)接口”,所以它可以使用Lambad表達(dá)式來(lái)代替匿名內(nèi)部類(lèi)。根據(jù)這個(gè)規(guī)則,我們?cè)囍鴣?lái)寫(xiě)一個(gè)函數(shù)接口,并使用Lambda表達(dá)式作為參數(shù)傳遞。
package com.coderbuff.custom;
/**
* 函數(shù)接口:只有一個(gè)方法的接口。作為L(zhǎng)ambda表達(dá)式的類(lèi)型
* Created by Kevin on 2018/2/17.
*/
public interface FunctionInterface {
void test();
}
測(cè)試:
package com.coderbuff.custom;
import org.junit.Test;
/**
* 函數(shù)接口測(cè)試
* Created by Kevin on 2018/2/17.
*/
public class FunctionInterfaceTest {
@Test
public void testLambda() {
func(new FunctionInterface() {
@Override
public void test() {
System.out.println("Hello World!");
}
});
//使用Lambda表達(dá)式代替上面的匿名內(nèi)部類(lèi)
func(() -> System.out.println("Hello World"));
}
private void func(FunctionInterface functionInterface) {
functionInterface.test();
}
}
可以看到,只要是一個(gè)接口中只包含一個(gè)方法,則可以使用Lambda表達(dá)式,這樣的接口稱(chēng)之為“函數(shù)接口”。
上面的函數(shù)接口比較簡(jiǎn)單不包含參數(shù),也不包含返回值。
我們?cè)賮?lái)修改FunctionInterface函數(shù)接口逐步加大Lambda表達(dá)式的難度——包含參數(shù),不包含返回值。
package com.coderbuff.custom;
/**
* 函數(shù)接口:只有一個(gè)方法的接口。作為L(zhǎng)ambda表達(dá)式的類(lèi)型
* Created by Kevin on 2018/2/17.
*/
public interface FunctionInterface {
void test(int param);
}
測(cè)試:
package com.coderbuff.custom;
import org.junit.Test;
/**
* 函數(shù)接口測(cè)試
* Created by Kevin on 2018/2/17.
*/
public class FunctionInterfaceTest {
@Test
public void testLambda() {
//使用Lambda表達(dá)式代替匿名內(nèi)部類(lèi)
func((x) -> System.out.println("Hello World" + x));
}
private void func(FunctionInterface functionInterface) {
int x = 1;
functionInterface.test(x);
}
}
關(guān)注Lambda表達(dá)式“(x)->Sysout.out.println("Hello World"+x)”,左邊傳遞的是參數(shù),此處并沒(méi)有指明參數(shù)類(lèi)型,因?yàn)樗梢酝ㄟ^(guò)上下文進(jìn)行類(lèi)型推導(dǎo),但在有些情況下不能推導(dǎo)出參數(shù)類(lèi)型(在編譯時(shí)不能推導(dǎo)通常IDE會(huì)提示),此時(shí)則需要指明參數(shù)類(lèi)型。我個(gè)人建議,任何情況下指明函數(shù)的參數(shù)類(lèi)型。
哪種情況不能推導(dǎo)出參數(shù)類(lèi)型呢?就是函數(shù)接口是一個(gè)泛型的時(shí)候。
package com.coderbuff.custom;
/**
* 函數(shù)接口:只有一個(gè)方法的接口。作為L(zhǎng)ambda表達(dá)式的類(lèi)型
* Created by Kevin on 2018/2/17.
*/
public interface FunctionInterface<T> {
void test(T param);
}
測(cè)試:
package com.coderbuff.custom;
import org.junit.Test;
/**
* 函數(shù)接口測(cè)試
* Created by Kevin on 2018/2/17.
*/
public class FunctionInterfaceTest {
@Test
public void testLambda() {
//使用Lambda表達(dá)式代替匿名內(nèi)部類(lèi)
func((Integer x) -> System.out.println("Hello World" + x));
}
private void func(FunctionInterface<Integer> functionInterface) {
int x = 1;
functionInterface.test(x);
}
}
上面的示例提到了Lambda表達(dá)式的兩種情況:
無(wú)參數(shù),無(wú)返回值;
有參數(shù),無(wú)返回值。
接下來(lái)就是有參數(shù),有返回值這種較為復(fù)雜的情況。
package com.coderbuff.custom;
/**
* 函數(shù)接口:只有一個(gè)方法的接口。作為L(zhǎng)ambda表達(dá)式的類(lèi)型
* Created by Kevin on 2018/2/17.
*/
public interface FunctionInterface<T> {
boolean test(T param);
}
測(cè)試:
package com.coderbuff.custom;
import org.junit.Test;
/**
* 函數(shù)接口測(cè)試
* Created by Kevin on 2018/2/17.
*/
public class FunctionInterfaceTest {
@Test
public void testLambda() {
//使用Lambda表達(dá)式代替匿名內(nèi)部類(lèi)
func((Integer x) -> true);
}
private void func(FunctionInterface<Integer> functionInterface) {
int x = 1;
functionInterface.test(x);
}
}
此時(shí)的Lambda表達(dá)式“(Integer x) -> true”,右邊是表達(dá)式的主體,直接返回true,如果有多行代碼,則可以直接使用花括號(hào)表示,例如:
func((Integer x) -> {
System.out.println("Hello World" + x);
return true;
});
Lambda表達(dá)式基本的語(yǔ)法規(guī)則:
無(wú)參數(shù),無(wú)返回值;
有參數(shù),無(wú)返回值;
有參數(shù),有返回值。
這三種基本情況已經(jīng)大致清楚了,特別是需要弄清,什么時(shí)候可以使用Lambda表達(dá)式代替匿名內(nèi)部類(lèi),也就是Lambda表達(dá)式的應(yīng)用場(chǎng)景是函數(shù)接口。Lambda表達(dá)式這一新特性在JDK8中的引入,更大的好處則是集合API的更新,新增的Stream類(lèi)庫(kù),使得我們?cè)诒闅v使用集合時(shí)不再像以往那樣不斷地使用for循環(huán)。
JDK8使用集合的正確姿勢(shì)
示例:計(jì)算來(lái)自“chengdu”的學(xué)生數(shù)量有多少。
在JDK8前的代碼:
for (Student student : studentList) {
if (student.getCity().equals("chengdu")) {
count++;
}
}
JDK8使用集合的正確姿勢(shì):
count = studentList.stream().filter((student -> student.getCity().equals("chengdu"))).count();
API的使用“難度”恰似提高了,實(shí)際只是不熟悉而已。傳統(tǒng)迭代的方式需要閱讀完整個(gè)循環(huán)才能明白代碼邏輯,JDK8通過(guò)流的方式則可以望文生義且代碼量大大減小。
以上就是動(dòng)力節(jié)點(diǎn)小編介紹的"JDK8的新特性詳解",希望對(duì)大家有幫助,想了解更多可查看Java8新特性技術(shù)文檔,如有疑問(wèn),請(qǐng)?jiān)诰€(xiàn)咨詢(xún),有專(zhuān)業(yè)老師隨時(shí)為您服務(wù)。
0基礎(chǔ) 0學(xué)費(fèi) 15天面授
有基礎(chǔ) 直達(dá)就業(yè)
業(yè)余時(shí)間 高薪轉(zhuǎn)行
工作1~3年,加薪神器
工作3~5年,晉升架構(gòu)
提交申請(qǐng)后,顧問(wèn)老師會(huì)電話(huà)與您溝通安排學(xué)習(xí)
初級(jí) 202925
初級(jí) 203221
初級(jí) 202629
初級(jí) 203743