什么時候可以使用 Lambda?通常 Lambda 表達式是用在函數式接口上使用的。從 Java8 開始引入了函數式接口,其說明比較簡單:函數式接口(Functional Interface)就是一個有且僅有一個抽象方法,但是可以有多個非抽象方法的接口。 java8 引入 @FunctionalInterface 注解聲明該接口是一個函數式接口。
一、語法抽象方法有且僅有一個
接口使用 @FunctionalInterface 注解進行標注
接口中可以存在默認方法和靜態方法實現
如下形式:
/**
*?定義函數式接口
*?接口上標注@FunctionalInterface?注解
*/
@FunctionalInterface
public?interface?ICollectionService{
/**
*?定義打印方法
*/
void?print();
}
在 Java8 以前,已有大量函數式接口形式的接口(接口中只存在一個抽象方法),只是沒有強制聲明。例如 java.lang.Runnable,java.util.concurrent.Callable,java.security.PrivilegedAction,java.io.FileFilter 等,Java8 新增加的函數接口在 java.util.function 包下,它包含了很多類,用來支持 Java 的函數式編程,該包中的函數式接口如下:
對于 Java8 中提供的這么多函數式接口,開發中常用的函數式接口有以下幾個 Predicate,Consumer,Function,Supplier。
二、函數式接口實例
2.1 Predicate
java.util.function.Predicate接口定義了一個名叫 test 的抽象方法,它接受泛型 T 對象,并返回一個 boolean 值。在對類型 T 進行斷言判斷時,可以使用這個接口。通常稱為斷言型接口 。「字符串判空」
Predicate?p01=(str)->str.isEmpty()||str.trim().isEmpty();
/**
*?測試傳入的字符串是否為空
*/
System.out.println(p01.test(""));
System.out.println(p01.test("?"));
System.out.println(p01.test("admin"));「用戶合法性校驗」
接口靜態方法完成手機號合法校驗功能,方法返回函數式接口 Predicate
public?interface?MyStringInter{
public?final??String?checkPhone=?"^((13[0-9])|(14[5,7,9])|(15([0-3]|[5-9]))|(16[0-9])"?+
"|(17[0,1,3,5,6,7,8])|(18[0-9])|(19[8|9]))\\d{8}$";
/**
*?用戶手機格式合法性
*?????返回L函數式接口Predicate?的實現?Lambda表達式
*?@return
*/
static?Predicate?checkPhone(){
return?(e)->?{
return?Pattern.compile(checkPhone).matcher(e).matches();
};
}
}
2.2 Consumer
java.util.function.Consumer接口定義了一個名叫 accept 的抽象方法,它接受泛型 T,沒有返回值(void)。如果需要訪問類型 T 的對象,并對其執行某些操作,可以使用這個接口,通常稱為消費型接口。「熱銷商品展示」
/**
熱銷商品測試數據
*/
Goods?g01=new?Goods(1,"iPad?2018款",3000,180,?BigDecimal.valueOf(2300));
Goods?g02=new?Goods(6,"小米平板4",5000,600,?BigDecimal.valueOf(1900));
Goods?g03=new?Goods(9,"微軟?Surface?Pro?6",100,50,?BigDecimal.valueOf(8500));
Goods?g04=new?Goods(20,"華為?榮耀平板5",1600,480,?BigDecimal.valueOf(1500));
List?goods=?Arrays.asList(g01,g02,g03,g04);
//Consumer?實現集合數據輸出?Lambda替代匿名函數?實現Consumer接口
goods.forEach(g->{
System.out.println(g);
});
2.3 Function
java.util.function.Function 接口定義了一個叫作 apply 的方法,它接受一個泛型 T 的對象,并返回一個泛型 R 的對象。如果需要定義一個 Lambda,將輸入的信息映射到輸出,可以使用這個接口(比如提取蘋果的重量,或把字符串映射為它的長度),通常稱為功能型接口。「用戶密碼 Base64 編碼」
//?實現用戶密碼?Base64加密操作
Function?f01=(password)->Base64.getEncoder().encodeToString(password.getBytes());
//?輸出加密后的字符串
System.out.println(f01.apply("123456"));
2.4 Supplier
java.util.function.Supplier接口定義了一個 get 的抽象方法,它沒有參數,返回一個泛型T的對象,這類似于一個工廠方法,通常稱為功能型接口。「外部Properties文件讀取」
public?static?Properties?readFile(String?fileName){
Supplier?supplier?=?()?->?{
try?{
InputStream?is?=?TestCase04.class.getClassLoader().getResourceAsStream(fileName);
Properties?prop?=?new?Properties();
prop.load(is);
return?prop;
}?catch?(IOException?e)?{
e.printStackTrace();
return?null;
}
};
return?supplier.get();
}
三、高階函數
Java8 中函數式接口中方法允許函數接口作為方法形參傳入,同時方法的結果為函數接口,從而實現鏈式調用操作,就像俄羅斯套娃那樣,當把套娃一個個打開時,發現還有一個同樣的小套娃在里面,最終發現最里面的一個也是一個完整的套娃玩具,此時的高階函數是不是跟套娃有著驚人的相似之處呢。「多頁面轉發」
String?action?=?"";
Predicate?p01?=?(a)?->?StringUtils.isBlank(a);
/**
*?如果action?為空?或?index?或?main?轉發到網站主頁面
*?鏈式判斷?方法結果仍然為一個函數
*/
if?(p01.or((a)?->?a.equals("index")).or((a)?->?a.equals("main")).test(action))?{
System.out.println("網站主頁面...");
}?else?{
System.out.println("其他頁面...");
}「多條件排序」
這里以商品數據為例,按商品銷量、評論排序,如果銷量一致按照商品評論數排序
/**
*?實際開發數據通常從數據庫獲取
*?這里使用測試數據
*/
Goods?g01=new?Goods(1,"小米9",1789,200,?BigDecimal.valueOf(2500));
Goods?g02=new?Goods(2,"華為Mate20",5000,3000,?BigDecimal.valueOf(7000));
Goods?g03=new?Goods(3,"OPPO?R17",2000,2827,?BigDecimal.valueOf(1500));
Goods?g04=new?Goods(4,"魅族?Note9",2000,1600,?BigDecimal.valueOf(1600));
Goods?g05=new?Goods(5,"一加6T",8000,5000,?BigDecimal.valueOf(3500));
List?goods=?Arrays.asList(g01,g02,g03,g04,g05);
//?銷量?與?評論排序?高階函數使用
Comparator?comparator?=?(g1,g2)->g1.getSale()-g2.getSale();
goods.sort(comparator.thenComparing(Comparator.comparing(g3?->?g3.getComment())));
goods.forEach((g)->System.out.println(g));
高階函數應用場景較多(這里查看源碼相關高階函數方法),如 Optinal 接口 filter、map 、orElseGet 等方法,Stream 流操作等基本都會用到 Predicate, Consumer, Supplier, Function 等接口。
四、函數式接口優勢與應用場景
函數式接口的引入,結合 Lambda 的使用,消除的匿名函數繁瑣的代碼,使得代碼結構簡潔、緊湊,第二點就是函數式接口中使用高階函數,可以很方便的實現鏈式調用,代碼清晰簡潔,同時引入的一種新的開發思想-函數式編程,對于開發者來說只需要關注函數的規則設計實現即可。
對于函數式接口應用,后續介紹到的 Optinal、Stream 相關方法對于數據處理的使用頻率較高,同時也是構成函數式編程的核心內容。