前言
在JDK8之前笼痹,Java是不支持函數(shù)式編程的酪穿,所謂的函數(shù)編程凳干,即可理解是將一個函數(shù)(也稱為“行為”)作為一個參數(shù)進行傳遞。通常我們提及得更多的是面向?qū)ο缶幊瘫患茫嫦驅(qū)ο缶幊淌菍?shù)據(jù)的抽象(各種各樣的POJO類)救赐,而函數(shù)式編程則是對行為的抽象(將行為作為一個參數(shù)進行傳遞)。
認識Lambda表達式
在IDEA中創(chuàng)建一個線程:
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("Hello World!");
}
});
DEA會給出提示可以使用Lambda表達式替換只磷。
使用Lambda表達式則只需要使用一句話就可代替上面使用匿名類的方式经磅。
new Thread(() -> System.out.println("Hello World!"));
在這個例子中,傳統(tǒng)的語法規(guī)則钮追,我們是將一個匿名內(nèi)部類作為參數(shù)進行傳遞预厌,我們實現(xiàn)了Runnable接口,并將其作為參數(shù)傳遞給Thread類畏陕,這實際上我們傳遞的是一段代碼配乓,也即我們將代碼作為了數(shù)據(jù)進行傳遞,這就帶來許多不必要的“樣板代碼”。
Lambda表達式一共有三部分組成:
什么時候在什么場景下可以使用Lambda表達式
能夠接收Lambda表達式的參數(shù)類型犹芹,是一個只包含一個方法的接口崎页。只包含一個方法的接口稱之為“函數(shù)接口”.
例如上面創(chuàng)建一個線程的示例,Runnable接口只包含一個方法腰埂,所以它被稱為“函數(shù)接口”飒焦,所以它可以使用Lambad表達式來代替匿名內(nèi)部類。
示例一(不包含參數(shù)屿笼,也不包含返回值的lambda表達式)
package com.coderbuff.custom;
/**
* 函數(shù)接口:只有一個方法的接口牺荠。作為Lambda表達式的類型
* Created by Kevin on 2018/2/17.
*/
public interface FunctionInterface {
void test();
}
package com.coderbuff.custom;
import org.junit.Test;
/**
* 函數(shù)接口測試
* Created by Kevin on 2018/2/17.
*/
public class FunctionInterfaceTest {
@Test
public void testLambda() {
func(new FunctionInterface() {
@Override
public void test() {
System.out.println("Hello World!");
}
});
//使用Lambda表達式代替上面的匿名內(nèi)部類
func(() -> System.out.println("Hello World"));
}
private void func(FunctionInterface functionInterface) {
functionInterface.test();
}
}
可以看到,只要是一個接口中只包含一個方法驴一,則可以使用Lambda表達式休雌,這樣的接口稱之為“函數(shù)接口”。
示例二 (包含參數(shù)肝断,不包含返回值的Lambda表達式)
package com.coderbuff.custom;
/**
* 函數(shù)接口:只有一個方法的接口杈曲。作為Lambda表達式的類型
* Created by Kevin on 2018/2/17.
*/
public interface FunctionInterface {
void test(int param);
}
package com.coderbuff.custom;
import org.junit.Test;
/**
* 函數(shù)接口測試
* Created by Kevin on 2018/2/17.
*/
public class FunctionInterfaceTest {
@Test
public void testLambda() {
//使用Lambda表達式代替匿名內(nèi)部類
func((x) -> System.out.println("Hello World" + x));
}
private void func(FunctionInterface functionInterface) {
int x = 1;
functionInterface.test(x);
}
}
Lambda表達式“(x) -> Sysout.out.println("Hello World" + x)”,左邊傳遞的是參數(shù)胸懈,此處并沒有指明參數(shù)類型担扑,因為它可以通過上下文進行類型推導(dǎo),但在有些情況下不能推導(dǎo)出參數(shù)類型(在編譯時不能推導(dǎo)通常IDE會提示)趣钱,此時則需要指明參數(shù)類型涌献。
哪種情況不能推導(dǎo)出參數(shù)類型呢?就是函數(shù)接口是一個泛型的時候首有。
package com.coderbuff.custom;
/**
* 函數(shù)接口:只有一個方法的接口燕垃。作為Lambda表達式的類型
* Created by Kevin on 2018/2/17.
*/
public interface FunctionInterface<T> {
void test(T param);
}
package com.coderbuff.custom;
import org.junit.Test;
/**
* 函數(shù)接口測試
* Created by Kevin on 2018/2/17.
*/
public class FunctionInterfaceTest {
@Test
public void testLambda() {
//使用Lambda表達式代替匿名內(nèi)部類
func((Integer x) -> System.out.println("Hello World" + x));
}
private void func(FunctionInterface<Integer> functionInterface) {
int x = 1;
functionInterface.test(x);
}
}
示例三(有參數(shù),有返回值的Lambda表達式)
package com.coderbuff.custom;
/**
* 函數(shù)接口:只有一個方法的接口绞灼。作為Lambda表達式的類型
* Created by Kevin on 2018/2/17.
*/
public interface FunctionInterface<T> {
boolean test(T param);
}
package com.coderbuff.custom;
import org.junit.Test;
/**
* 函數(shù)接口測試
* Created by Kevin on 2018/2/17.
*/
public class FunctionInterfaceTest {
@Test
public void testLambda() {
//使用Lambda表達式代替匿名內(nèi)部類
func((Integer x) -> true);
}
private void func(FunctionInterface<Integer> functionInterface) {
int x = 1;
functionInterface.test(x);
}
}
此時的Lambda表達式“(Integer x) -> true”利术,右邊是表達式的主體,直接返回true低矮,如果有多行代碼印叁,則可以直接使用花括號表示,例如:
func((Integer x) -> {
System.out.println("Hello World" + x);
return true;
});
**這三種基本情況已經(jīng)大致清楚了军掂,特別是需要弄清轮蜕,什么時候可以使用Lambda表達式代替匿名內(nèi)部類,也就是Lambda表達式的應(yīng)用場景是函數(shù)接口蝗锥。Lambda表達式這一新特性在JDK8中的引入跃洛,更大的好處則是集合API的更新,新增的Stream類庫终议,使得我們在遍歷使用集合時不再像以往那樣不斷地使用for循環(huán)汇竭。
常用的函數(shù)式接口
@FunctionalInterface注解
@FunctionalInterface注解的功能就是約束一個接口只能有一個方法葱蝗,參數(shù)沒有限制,返回值可有可無
當一個接口使用該注解细燎,就代表該接口是一個函數(shù)式接口两曼,也就可以使用lambda表達式簡化使用
四大函數(shù)式接口
- ① Function<T, R> : 函數(shù)型接口
有參數(shù)、有返回值
R apply(T t);
- ② Consumer : 消費型接口
有參數(shù)玻驻,無返回值
void accept(T t);
- ③ Supplier : 供給型接口
無參數(shù)悼凑,有返回值
T get();
- ④ Predicate : 判斷型接口
有參數(shù),返回一個boolean值
boolean test(T t);