作為老一代Java工作者,已經(jīng)習(xí)慣了寫匿名類,在Java代碼中除了基本的數(shù)據(jù)類型楞抡,一切的都是對象家厌,定義的函數(shù)和方法不可能完全獨(dú)立,也不能將方法作為參數(shù)或者返回一個方法給實(shí)例。但是我們經(jīng)常通過匿名類給方法傳遞函數(shù)功能,比如 下面代碼
textview.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//todo something
}
});
在函數(shù)式編程語言中,函數(shù)它們可以獨(dú)立存在邻吞,你可以將其賦值給一個變量,或?qū)⑺麄儺?dāng)做參數(shù)傳給其他函數(shù)葫男。
但是在Java中我們不能這么做抱冷,所以就出現(xiàn)了Lambda表達(dá)式來彌補(bǔ)這缺陷,Lambda 表達(dá)式為 Java 添加了缺失的函數(shù)式編程特點(diǎn)腾誉。
Lambda 表達(dá)式簡介
Lambda 表達(dá)式是一種匿名函數(shù)(對 Java 而言這并不完全正確徘层,但現(xiàn)在姑且這么認(rèn)為),簡單地說利职,它是沒有聲明的方法趣效,也即沒有訪問修飾符、返回值聲明和名字猪贪。
你可以將其想做一種速記跷敬,在你需要使用某個方法的地方寫上它。當(dāng)某個方法只使用一次热押,而且定義很簡短西傀,使用這種速記替代之尤其有效斤寇,這樣,你就不必在類中費(fèi)力寫聲明與方法了拥褂。
Lambda語法
Lambda 表達(dá)式通常表示為如下格式:
(參數(shù)) ->{處理代碼}
比如如下:
(int a, int b )->{return a*b}
() ->{return "test"}
....
必須了解的基本規(guī)則
- Lambda 表達(dá)式可以有多個或者零個參數(shù)
- 參數(shù)類型 可以不申明 (int a)與 (a)效果相同
- 參數(shù)用逗號相隔 (int a, int b, int c)
- 只有一個參數(shù)括號可以省略 a -> return a;
- {} 內(nèi)可以有代碼塊 a ->{ int b = 2; int c = a+b; return c;}
函數(shù)式接口
只包含一個抽象方法聲明的接口叫函數(shù)式接口娘锁。
java.lang.Runnable 就是一種函數(shù)式接口,在 Runnable 接口中只聲明了一個方法 void run()饺鹃;
簡單的說就是莫秆,寫成匿名內(nèi)部類的時候,只有一個內(nèi)部方法的接口悔详。我們可以看看通過Lambda表達(dá)式創(chuàng)建Runnable 接口引用镊屎。
Runnable r = () -> System.out.println("hello world");
new Thread(r);
or
new Thread(
() -> System.out.println("hello world")
).start();
Thread 構(gòu)造函數(shù)將Lambda表達(dá)式作為參數(shù)傳給了類對象。
FunctionalInterface
@FunctionalInterface 是 Java 8 新加入的一種接口茄螃,用于指明該接口類型聲明是根據(jù) Java 語言規(guī)范定義的函數(shù)式接口缝驳。Java 8 還聲明了一些 Lambda 表達(dá)式可以使用的函數(shù)式接口,當(dāng)你注釋的接口不是有效的函數(shù)式接口時归苍,可以使用 @FunctionalInterface 解決編譯層面的錯誤用狱。比如:
@FunctionalInterface
public interface TestInterface {
public void test();
}
//根據(jù)定義,函數(shù)式接口只能有一個抽象方法霜医,如果你嘗試添加第二
//個抽象方法齿拂,將拋出編譯時錯誤驳规。例如:
@FunctionalInterface
public interface TestInterface {
public void test1();
public void test2();
}
我們看一下使用方法
public class Test{
public void run( TestInterface test){
test.test();
}
//傳統(tǒng)用法
run(new TestInterface(){
@Override
public void test() {
//do something
System.out.println(" Anonymous class");
}
});
//Lambda
run( () ->System.out.println(" Lambda expression"));
}
//舊方法:
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("Hello from thread");
}
}).start();
//新方法:
new Thread(
() -> System.out.println("Hello from thread")
).start();
全新的操作符雙冒號(::)
Java 8 更新了一個新的操作符:: 雙冒號操作符可以講一個常規(guī)的方法轉(zhuǎn)換為Lambda表達(dá)式:
//Old way:
List<Integer> list = Arrays.asList(1, 2, 3, 4, 5, 6, 7);
for(Integer n: list) {
System.out.println(n);
}
//New way:
List<Integer> list = Arrays.asList(1, 2, 3, 4, 5, 6, 7);
list.forEach(n -> System.out.println(n));
//or we can use :: double colon operator in Java 8
list.forEach(System.out::println);
Lambda 表達(dá)式與匿名類的區(qū)別
使用匿名類與 Lambda 表達(dá)式的一大區(qū)別在于關(guān)鍵詞的使用肴敛。對于匿名類,關(guān)鍵詞 this 解讀為匿名類吗购,而對于 Lambda 表達(dá)式医男,關(guān)鍵詞 this 解讀為寫就 Lambda 的外部類。
寫文不容易捻勉,點(diǎn)個贊唄镀梭。