標籤:
函數式介面
函數式介面(functional interface 也叫功能性介面,其實是同一個東西)。簡單來說,函數式介面是只包含一個方法的介面。比如Java標準庫中的java.lang.Runnable和 java.util.Comparator都是典型的函數式介面。java 8提供 @FunctionalInterface作為註解,這個註解是非必須的,只要介面符合函數式介面的標準(即只包含一個方法的介面),虛擬機器會自動判斷, 但 最好在介面上使用註解@FunctionalInterface進行聲明,以免團隊的其他人員錯誤地往介面中添加新的方法。
Java中的lambda無法單獨出現,它需要一個函數式介面來盛放,lambda運算式方法體其實就是函數介面的實現,下面講到文法會講到 。
Lambda文法
包含三個部分
一個括弧內用逗號分隔的形式參數,參數是函數式介面裡面方法的參數
一個箭頭符號:->
方法體,可以是運算式和代碼塊,方法體函數式介面裡面方法的實現,如果是代碼塊,則必須用{}來包裹起來,且需要一個return 傳回值,但有個例外,若函數式介面裡面方法傳回值是void,則無需{}
public class TestLambda { public static void runThreadUseLambda() { //Runnable是一個函數介面,只包含了有個無參數的,返回void的run方法; //所以lambda運算式左邊沒有參數,右邊也沒有return,只是單純的列印一句話 new Thread(() ->System.out.println("lambda實現的線程")).start(); } public static void runThreadUseInnerClass() { //這種方式就不多講了,以前舊版本比較常見的做法 new Thread(new Runnable() { @Override public void run() { System.out.println("內部類實現的線程"); } }).start(); } public static void main(String[] args) { TestLambda.runThreadUseLambda(); TestLambda.runThreadUseInnerClass(); }}
可以看出,使用lambda運算式設計的代碼會更加簡潔,而且還可讀。
方法引用
其實是lambda運算式的一個簡化寫法,所引用的方法其實是lambda運算式的方法體實現,文法也很簡單,左邊是容器(可以是類名,執行個體名),中間是"::",右邊是相應的方法名。
一般方法的引用格式是
如果是靜態方法,則是ClassName::methodName。如 Object ::equals
如果是執行個體方法,則是Instance::methodName。如Object obj=new Object();obj::equals;
建構函式.則是ClassName::new
再來看一個完整的例子,方便理解
import java.awt.FlowLayout;import java.awt.event.ActionEvent;import javax.swing.JButton;import javax.swing.JFrame; public class TestMethodReference { public static void main(String[] args) { JFrame frame = new JFrame(); frame.setLayout(new FlowLayout()); frame.setVisible(true); JButton button1 = new JButton("點擊我!"); JButton button2 = new JButton("點擊我!"); frame.getContentPane().add(button1); frame.getContentPane().add(button2); //這裡addActionListener方法的參數是ActionListener,是一個函數式介面 //使用lambda運算式方式 button1.addActionListener(e -> { System.out.println("這裡是Lambda實現方式"); }); //使用方法引用方式 button2.addActionListener(TestMethodReference::doSomething); } /** * 這裡是函數式介面ActionListener的實現方法 * @param e */ public static void doSomething(ActionEvent e) { System.out.println("這裡是方法引用實現方式"); }}
為什麼需要Lambda運算式
主要有三個原因:
> 更加緊湊的代碼
比如Java中現有的匿名內部類以及監聽器(listeners)和事件處理器(handlers)都顯得很冗長
> 修改方法的能力(我個人理解為代碼注入,或者有點類似JavaScript中傳一個回呼函數給另外一個函數)
比如Collection介面的contains方法,若且唯若傳入的元素真正包含在集合中,才返回true。而假如我們想對一個字串集合,傳入一個字串,只要這個字串出現在集合中(忽略大小寫)就返回true。
簡單地說,我們想要的是傳入“一些我們自己的代碼”到已有的方法中,已有的方法將會執行我們傳入的代碼。Lambda運算式能很好地支援這點
> 更好地支援多核處理
例如,通過Java 8新增的Lambda運算式,我們可以很方便地並行操作大集合,充分發揮多核CPU的潛能。
平行處理函數如filter、map和reduce。
Java基礎知識總結之1.8新特性lambda運算式