說到java 8,首先會想到lambda(閉包)以及虛擬擴展方法(default method),這個特性早已經(jīng)被各大技術網(wǎng)站炒得沸沸揚揚了,也是我們java 8系列開篇要講的第一特性(JEP126 http://openjdk.java.net/jeps/126),jdk8的一些庫已經(jīng)應用了lambda表達式重新設計了,理解他對學習java 8新特性有著重要的意義。
一、函數(shù)式接口
函數(shù)式接口(functional interface 也叫功能性接口,其實是同一個東西)。簡單來說,函數(shù)式接口是只包含一個方法的接口。比如Java標準庫中的java.lang.Runnable和java.util.Comparator都是典型的函數(shù)式接口。java 8提供 @FunctionalInterface作為注解,這個注解是非必須的,只要接口符合函數(shù)式接口的標準(即只包含一個方法的接口),虛擬機會自動判斷,但 最好在接口上使用注解@FunctionalInterface進行聲明,以免團隊的其他人員錯誤地往接口中添加新的方法。
Java中的lambda無法單獨出現(xiàn),它需要一個函數(shù)式接口來盛放,lambda表達式方法體其實就是函數(shù)接口的實現(xiàn),下面講到語法會講到
二、Lambda語法
包含三個部分
1.一個括號內(nèi)用逗號分隔的形式參數(shù),參數(shù)是函數(shù)式接口里面方法的參數(shù)
2.一個箭頭符號:->
3.方法體,可以是表達式和代碼塊,方法體函數(shù)式接口里面方法的實現(xiàn),如果是代碼塊,則必須用{}來包裹起來,且需要一個return 返回值,但有個例外,若函數(shù)式接口里面方法返回值是void,則無需{}
總體看起來像這樣:
(parameters) -> expression 或者 (parameters) -> { statements; }
看一個完整的例子,方便理解
/**
* 測試lambda表達式
*
* @author benhail
*/
public class TestLambda {
public static void runThreadUseLambda() {
//Runnable是一個函數(shù)接口,只包含了有個無參數(shù)的,返回void的run方法;
//所以lambda表達式左邊沒有參數(shù),右邊也沒有return,只是單純的打印一句話
new Thread(() ->System.out.println("lambda實現(xiàn)的線程")).start();
}
public static void runThreadUseInnerClass() {
//這種方式就不多講了,以前舊版本比較常見的做法
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("內(nèi)部類實現(xiàn)的線程");
}
}).start();
}
public static void main(String[] args) {
TestLambda.runThreadUseLambda();
TestLambda.runThreadUseInnerClass();
}
}
可以看出,使用lambda表達式設計的代碼會更加簡潔,而且還可讀。
三、方法引用
其實是lambda表達式的一個簡化寫法,所引用的方法其實是lambda表達式的方法體實現(xiàn),語法也很簡單,左邊是容器(可以是類名,實例名),中間是"::",右邊是相應的方法名。如下所示:
一般方法的引用格式是
如果是靜態(tài)方法,則是ClassName::methodName。如 Object ::equals
如果是實例方法,則是Instance::methodName。如Object obj=new Object();obj::equals;
構造函數(shù).則是ClassName::new
再來看一個完整的例子,方便理解:
import java.awt.FlowLayout;
import java.awt.event.ActionEvent;
import javax.swing.JButton;
import javax.swing.JFrame;
/**
*
* @author benhail
*/
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方法的參數(shù)是ActionListener,是一個函數(shù)式接口
//使用lambda表達式方式
button1.addActionListener(e -> { System.out.println("這里是Lambda實現(xiàn)方式"); });
//使用方法引用方式
button2.addActionListener(TestMethodReference::doSomething);
}
/**
* 這里是函數(shù)式接口ActionListener的實現(xiàn)方法
* @param e
*/
public static void doSomething(ActionEvent e) {
System.out.println("這里是方法引用實現(xiàn)方式");
}
}
可以看出,doSomething方法就是lambda表達式的實現(xiàn),這樣的好處就是,如果你覺得lambda的方法體會很長,影響代碼可讀性,方法引用就是個解決辦法
四、總結
以上就是lambda表達式語法的全部內(nèi)容了,相信大家對lambda表達式都有一定的理解了,但只是代碼簡潔了這個好處的話,并不能打動很多觀眾,java 8也不會這么令人期待,其實java 8引入lambda迫切需求是因為lambda 表達式能簡化集合上數(shù)據(jù)的多線程或者多核的處理,提供更快的集合處理速度 ,這個后續(xù)會講到,關于JEP126的這一特性,將分3部分,之所以分開,是因為這一特性可寫的東西太多了,這部分讓讀者熟悉lambda表達式以及方法引用的語法和概念,第二部分則是虛擬擴展方法(default method)的內(nèi)容,最后一部分則是大數(shù)據(jù)集合的處理,解開lambda表達式的最強作用的神秘面紗。敬請期待。。。。