Java8被稱(chēng)作Java史上變化最大的一個(gè)版本。其中包含很多重要的新特性,最核心的就是增加了Lambda表達(dá)式和StreamAPI。這兩者也可以結(jié)合在一起使用。首先來(lái)看下什么是Lambda表達(dá)式。
使用Lambda表達(dá)式不僅讓代碼變的簡(jiǎn)單、而且可讀、最重要的是代碼量也隨之減少很多。然而,在某種程度上,這些功能在Scala等這些JVM語(yǔ)言里已經(jīng)被廣泛使用。
并不奇怪,Scala社區(qū)是難以置信的,因?yàn)樵S多Java 8里的內(nèi)容看起來(lái)就像是從Scala里搬過(guò)來(lái)的。在某種程度上,Java 8的語(yǔ)法要比Scala的更詳細(xì)但不是很清晰,但這并不能說(shuō)明什么,如果可以,它可能會(huì)像Scala那樣構(gòu)建Lambda表達(dá)式。
一方面,如果Java繼續(xù)圍繞Lambda來(lái)發(fā)展和實(shí)現(xiàn)Scala都已經(jīng)實(shí)現(xiàn)的功能,那么可能就不需要Scala了。另一方面,如果它只提供一些核心的功能,例如幫助匿名內(nèi)部類(lèi),那么Scala和其他語(yǔ)言將會(huì)繼續(xù)茁壯成長(zhǎng),并且有可能會(huì)凌駕于Java之上。其實(shí)這才是最好的結(jié)果,有競(jìng)爭(zhēng)才有進(jìn)步,其它語(yǔ)言繼續(xù)發(fā)展和成長(zhǎng),并且無(wú)需擔(dān)心是否會(huì)過(guò)時(shí)。
Lambda表達(dá)式,維基百科上的解釋是一種用于表示匿名函數(shù)和閉包的運(yùn)算符,感覺(jué)看到這個(gè)解釋還是覺(jué)得很抽象,接下來(lái)我們看一個(gè)例子
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
public class SwingTest { public static void main(String[] args) { JFrame jFrame = new JFrame( "My JFrame" ); JButton jButton = new JButton( "My JButton" ); jButton.addActionListener( new ActionListener() { @Override public void actionPerformed(ActionEvent e) { System.out.println( "Button Pressed!" ); } }); jFrame.add(jButton); jFrame.pack(); jFrame.setVisible( true ); jFrame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); } } |
這是一段Swing編程中的代碼,給Button綁定一個(gè)監(jiān)聽(tīng)事件,當(dāng)點(diǎn)擊Button時(shí)會(huì)在控制臺(tái)輸出"ButtonPressed!"內(nèi)容。這里使用了創(chuàng)建了一個(gè)匿名內(nèi)部類(lèi)的實(shí)例來(lái)綁定到監(jiān)聽(tīng)器,這也是以往比較常規(guī)的代碼組織形式。但是仔細(xì)看一下我們會(huì)發(fā)現(xiàn),實(shí)際上我們真正關(guān)注的就是一個(gè)ActionEvent類(lèi)型的參數(shù)e和向控制臺(tái)輸出的語(yǔ)句System.out.println("ButtonPressed!");。
如果將上段程序中以匿名內(nèi)部類(lèi)的方式創(chuàng)建接口實(shí)例的代碼替換成Lambda表達(dá)式后,代碼如下
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
public class SwingTest { public static void main(String[] args) { JFrame jFrame = new JFrame( "My JFrame" ); JButton jButton = new JButton( "My JButton" ); jButton.addActionListener(e -> System.out.println( "Button Pressed!" )); jFrame.add(jButton); jFrame.pack(); jFrame.setVisible( true ); jFrame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); } } |
關(guān)注最中間部分代碼的變化,由原來(lái)的6行代碼,現(xiàn)在1行就可以實(shí)現(xiàn)了。這就是Lambda表達(dá)式的一種簡(jiǎn)單形式。
可以看出Lambda表達(dá)式的語(yǔ)法是
1
2
3
4
|
(param1,param2,param3) -> { //todo } |
這里參數(shù)的類(lèi)型程序可以根據(jù)上下文進(jìn)行推斷,但是并不是所有的類(lèi)型都可以推斷出來(lái),此時(shí)就需要我們顯示的聲明參數(shù)類(lèi)型,當(dāng)只有一個(gè)參數(shù)時(shí)小括號(hào)可以省略。當(dāng)todo部分只有一行代碼時(shí),外邊的大括號(hào)可以省略。如我們上面的示例
那么除了代碼簡(jiǎn)潔了,Lambda表達(dá)式還給我們帶來(lái)了什么變化嗎?
我們回憶一下,在Java中,我們是否無(wú)法將函數(shù)作為參數(shù)傳遞給一個(gè)方法,也無(wú)法聲明返回值是一個(gè)函數(shù)的方法。在Java8之前,答案是肯定的。
那么,在上面的例子中我們居然可以將一段代碼邏輯作為參數(shù)傳遞給了監(jiān)聽(tīng)器,告訴監(jiān)聽(tīng)器事件觸發(fā)時(shí)你可以這么做,而不再需要以匿名內(nèi)部類(lèi)的方式作為參數(shù)。這也是Java8帶來(lái)的另一新特性:函數(shù)式編程。
支持函數(shù)式編程的語(yǔ)言有很多,在JavaScript中,把函數(shù)作為參數(shù)傳遞,或者返回值是一個(gè)函數(shù)的情況非常常見(jiàn),JavaScript是一門(mén)非常常見(jiàn)的函數(shù)式語(yǔ)言。
Lambda為Java添加了缺失的函數(shù)式編程的特性,使我們能將函數(shù)當(dāng)做一等公民看待。
在函數(shù)式編程語(yǔ)言中,Lambda表達(dá)式的類(lèi)型是函數(shù)。而在Java中,Lambda表達(dá)式是對(duì)象,它們必須依附于一類(lèi)特別的對(duì)象類(lèi)型——函數(shù)式接口(FunctionalInterface)。
接下來(lái)我們看下函數(shù)式接口的定義:
如果一個(gè)接口中,有且只有一個(gè)抽象的方法(Object類(lèi)中的方法不包括在內(nèi)),那這個(gè)接口就可以被看做是函數(shù)式接口。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
@FunctionalInterface public interface Runnable { /** * When an object implementing interface <code>Runnable</code> is used * to create a thread, starting the thread causes the object's * <code>run</code> method to be called in that separately executing * thread. * <p> * The general contract of the method <code>run</code> is that it may * take any action whatsoever. * * @see java.lang.Thread#run() */ public abstract void run(); } |
來(lái)看下Runnable接口的聲明,在Java8后,Runnable接口多了一個(gè)FunctionalInterface注解,表示該接口是一個(gè)函數(shù)式接口。但是如果我們不添加FunctionalInterface注解的話,如果接口中有且只有一個(gè)抽象方法時(shí),編譯器也會(huì)把該接口當(dāng)做函數(shù)式接口看待。
1
2
3
4
5
|
@FunctionalInterface public interface MyInterface { void test(); String toString(); } |
MyInterface這也是一個(gè)函數(shù)式接口,因?yàn)閠oString()是Object類(lèi)中的方法,只是在這里進(jìn)行了復(fù)寫(xiě),不會(huì)增加接口中抽象方法的數(shù)量。
(到這里額外提一下,Java8中,接口里面的方法不僅僅只能有抽象方法,也可以有具體實(shí)現(xiàn)了的方法,被稱(chēng)作默認(rèn)方法(defaultmethod),這部分后面會(huì)具體介紹)
既然在Java中,Lambda表達(dá)式是對(duì)象。那么這個(gè)對(duì)象的類(lèi)型是什么呢?我們?cè)倩仡櫹耂wingTest程序,這里以匿名內(nèi)部類(lèi)的方式創(chuàng)建了一個(gè)ActionListener接口實(shí)例
1
2
3
4
5
6
|
jButton.addActionListener( new ActionListener() { @Override public void actionPerformed(ActionEvent e) { System.out.println( "Button Pressed!" ); } }); |
使用Lambda表達(dá)式改進(jìn)后
1
|
jButton.addActionListener(e -> System.out.println( "Button Pressed!" )); |
也就是我們使用Lambda表達(dá)式創(chuàng)建了一個(gè)ActionListener接口的實(shí)例,再看下ActionListener接口的定義
1
2
3
4
5
6
|
public interface ActionListener extends EventListener { /** * Invoked when an action occurs. */ public void actionPerformed(ActionEvent e); } |
只有一個(gè)抽象方法,雖然沒(méi)添加FunctionalInterface注解,但是也符合函數(shù)式接口的定義,編譯器會(huì)認(rèn)為這是一個(gè)函數(shù)式接口。
所以,使用Lambda表達(dá)式可以創(chuàng)建函數(shù)式接口的實(shí)例。即Lambda表達(dá)式返回的是函數(shù)式接口類(lèi)型。
實(shí)際上,函數(shù)式接口實(shí)例的創(chuàng)建可以有三種方式(參考自FunctionalInterface注解說(shuō)明):
1.Lambda表達(dá)式
2.方法引用
3.構(gòu)造方法引用
總結(jié)
以上就是本文關(guān)于Java8簡(jiǎn)單了解Lambda表達(dá)式與函數(shù)式接口的全部?jī)?nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站其他相關(guān)專(zhuān)題,如有不足之處,歡迎留言指出!
原文鏈接:https://segmentfault.com/a/1190000012211339