摘要:而在中,表達式是對象,它們必須依附于一類特別的對象類型函數式接口。即表達式返回的是函數式接口類型。
Java8被稱作Java史上變化最大的一個版本。其中包含很多重要的新特性,最核心的就是增加了Lambda表達式和Stream API。這兩者也可以結合在一起使用。首先來看下什么是Lambda表達式。
Lambda表達式,維基百科上的解釋是一種用于表示匿名函數和閉包的運算符,感覺看到這個解釋還是覺得很抽象,接下來我們看一個例子
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綁定一個監聽事件,當點擊Button時會在控制臺輸出"Button Pressed!"內容。這里使用了創建了一個匿名內部類的實例來綁定到監聽器,這也是以往比較常規的代碼組織形式。但是仔細看一下我們會發現,實際上我們真正關注的就是一個ActionEvent類型的參數e和向控制臺輸出的語句System.out.println("Button Pressed!");。
如果將上段程序中以匿名內部類的方式創建接口實例的代碼替換成Lambda表達式后,代碼如下
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); }
}
關注最中間部分代碼的變化,由原來的6行代碼,現在1行就可以實現了。這就是Lambda表達式的一種簡單形式。
可以看出Lambda表達式的語法是
(param1,param2,param3) -> {
//todo
}
這里參數的類型程序可以根據上下文進行推斷,但是并不是所有的類型都可以推斷出來,此時就需要我們顯示的聲明參數類型,當只有一個參數時小括號可以省略。當todo部分只有一行代碼時,外邊的大括號可以省略。如我們上面的示例
那么除了代碼簡潔了,Lambda表達式還給我們帶來了什么變化嗎?
我們回憶一下,在Java中,我們是否無法將函數作為參數傳遞給一個方法,也無法聲明返回值是一個函數的方法。在Java8之前,答案是肯定的。
那么,在上面的例子中我們居然可以將一段代碼邏輯作為參數傳遞給了監聽器,告訴監聽器事件觸發時你可以這么做,而不再需要以匿名內部類的方式作為參數。這也是Java8帶來的另一新特性:函數式編程。
支持函數式編程的語言有很多,在JavaScript中,把函數作為參數傳遞,或者返回值是一個函數的情況非常常見,JavaScript是一門非常常見的函數式語言。
Lambda為Java添加了缺失的函數式編程的特性,使我們能將函數當做一等公民看待。
在函數式編程語言中,Lambda表達式的類型是函數。而在Java中,Lambda表達式是對象,它們必須依附于一類特別的對象類型——函數式接口(Functional Interface)。
接下來我們看下函數式接口的定義:
如果一個接口中,有且只有一個抽象的方法(Object類中的方法不包括在內),那這個接口就可以被看做是函數式接口。
@FunctionalInterface public interface Runnable { /** * When an object implementing interfaceRunnable
is used * to create a thread, starting the thread causes the object"s *run
method to be called in that separately executing * thread. ** The general contract of the method
run
is that it may * take any action whatsoever. * * @see java.lang.Thread#run() */ public abstract void run(); }
來看下Runnable接口的聲明,在Java8后,Runnable接口多了一個FunctionalInterface注解,表示該接口是一個函數式接口。但是如果我們不添加FunctionalInterface注解的話,如果接口中有且只有一個抽象方法時,編譯器也會把該接口當做函數式接口看待。
@FunctionalInterface public interface MyInterface { void test(); String toString(); }
MyInterface這也是一個函數式接口,因為toString()是Object類中的方法,只是在這里進行了復寫,不會增加接口中抽象方法的數量。
(到這里額外提一下,Java8中,接口里面的方法不僅僅只能有抽象方法,也可以有具體實現了的方法,被稱作默認方法(default method),這部分后面會具體介紹)
既然在Java中,Lambda表達式是對象。那么這個對象的類型是什么呢?我們再回顧下SwingTest程序,這里以匿名內部類的方式創建了一個ActionListener接口實例
jButton.addActionListener(new ActionListener() { @Override public void actionPerformed(ActionEvent e) { System.out.println("Button Pressed!"); } });
使用Lambda表達式改進后
jButton.addActionListener(e -> System.out.println("Button Pressed!"));
也就是我們使用Lambda表達式創建了一個ActionListener接口的實例,再看下ActionListener接口的定義
public interface ActionListener extends EventListener { /** * Invoked when an action occurs. */ public void actionPerformed(ActionEvent e); }
只有一個抽象方法,雖然沒添加FunctionalInterface注解,但是也符合函數式接口的定義,編譯器會認為這是一個函數式接口。
所以,使用Lambda表達式可以創建函數式接口的實例。即Lambda表達式返回的是函數式接口類型。
實際上,函數式接口實例的創建可以有三種方式(參考自FunctionalInterface注解說明):
1.Lambda表達式
2.方法引用(后續章節介紹)
3.構造方法引用(后續章節介紹)
小結:本篇我們打開了學習Java8的大門,認識了什么是lambda表達式,了解了函數式接口的定義是什么,并借住幾個例子展示了lambda表達式的便捷之處,如果覺得本篇文章對你有所幫助幫忙贊一下哈。
下一篇
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/70653.html
摘要:看起來好像是廢話,它還有一個補充的說明,在函數式編程中要避免狀態變化和使用可變對象。函數式編程的特點在中,函數即對象,例如聲明一個函數之后,你可以調用其屬性。 ...
摘要:表達式的主要作用就是代替匿名內部類的煩瑣語法。從這點來看,表達式的代碼塊與匿名內部類的方法體是相同的。與匿名內部類相似的是,由于表達式訪問了局部變量,該局部變量相當于與一個隱式的修飾,因此不允許對局部變量重新賦值。 函數式接口 函數式接口(Functional Interface)就是一個只有一個抽象方法(可以包含多個默認方法或多個static方法)的普通接口,可以被隱式轉換為lamb...
摘要:在支持一類函數的語言中,表達式的類型將是函數。匿名函數的返回類型與該主體表達式一致如果表達式的主體包含一條以上語句,則表達式必須包含在花括號中形成代碼塊。注意,使用表達式的方法不止一種。 摘要:此篇文章主要介紹 Java8 Lambda 表達式產生的背景和用法,以及 Lambda 表達式與匿名類的不同等。本文系 OneAPM 工程師編譯整理。 Java 是一流的面向對象語言,除了部分簡...
摘要:利用前面所述的方法,這個例子可以用方法引用改寫成下面的樣子構造函數引用對于一個現有構造函數,你可以利用它的名稱和關鍵字來創建它的一個引用。 第三章 Lambda表達式 函數式接口 函數式接口就是只定義一個抽象方法的接口,哪怕有很多默認方法,只要接口只定義了一個抽象方法,它就仍然是一個函數式接口。 常用函數式接口 showImg(https://segmentfault.com/img...
摘要:函數式接口可以包含多個默認方法類方法但只能一個抽象方法函數式接口有且只有一個沒有關鍵字修飾的接口可以多個關鍵字的抽象方法類方法變量語法組成元素形參列表。 Lambda相當于就是一個匿名方法,其在代替匿名內部類創建對象的時候,Lambda表達式代碼塊會代替實現抽象方法的方法體 Lambda表達式的目標類型必須是函數式接口(FunctionalInterface)。函數式接口只能包含一個抽...
閱讀 2864·2021-11-16 11:55
閱讀 2608·2021-09-29 09:34
閱讀 3405·2021-09-01 14:21
閱讀 3753·2019-08-29 12:36
閱讀 697·2019-08-26 10:55
閱讀 3959·2019-08-26 10:20
閱讀 1026·2019-08-23 18:19
閱讀 1194·2019-08-23 17:56