
## 一、函數式接口是什么?
所謂的函數式接口,實際上就是接口里面**只能有一個抽象方法的接口**。我們上一節用到的Comparator接口就是一個典型的函數式接口,它只有一個抽象方法compare。

只有一個抽象方法?那上圖中的equals方法不是也沒有函數體么?不急,和我一起往下看!
## 二、函數式接口的特點
* 接口有且僅有一個抽象方法,如上圖的抽象方法compare
* 允許定義靜態非抽象方法
* 允許定義默認defalut非抽象方法(default方法也是java8才有的,見下文)
* 允許java.lang.Object中的public方法,如上圖的方法equals。
* FunctionalInterface注解不是必須的,如果一個接口符合"函數式接口"定義,那么加不加該注解都沒有影響。加上該注解能夠更好地讓編譯器進行檢查。如果編寫的不是函數式接口,但是加上了@FunctionalInterface,那么編譯器會報錯
甚至可以說:函數式接口是專門為lambda表達式準備的,**lambda表達式是只實現接口中唯一的抽象方法的匿名實現類**。
## 三、default關鍵字
順便講一下default關鍵字,在java8之前
* 接口是不能有方法的實現,所有方法全都是抽象方法
* 實現接口就必須實現接口里面的所有方法
這就導致一個問題:**當一個接口有很多的實現類的時候,修改這個接口就變成了一個非常麻煩的事,需要修改這個接口的所有實現類**。
這個問題困擾了java工程師許久,不過在java8中這個問題得到了解決,沒錯就是default方法
* default方法可以有自己的默認實現,即有方法體。
* 接口實現類可以不去實現default方法,并且可以使用default方法。
## 四、JDK中的函數式接口舉例
java.lang.Runnable,
java.util.Comparator,
java.util.concurrent.Callable
java.util.function包下的接口,如Consumer、Predicate、Supplier等
## 五、自定義Comparator排序
我們自定義一個排序器,實現compare函數(函數式接口Comparator唯一的抽象方法)。返回0表示元素相等,-1表示前一個元素小于后一個元素,1表示前一個元素大于后一個元素。這個規則和java 8之前沒什么區別。
下面代碼用自定義接口實現類的的方式實現:按照年齡的倒序排序!
~~~
employees.sort(new Comparator<Employee>() {
@Override
public int compare(Employee em1, Employee em2) {
if(em1.getAge() == em2.getAge()){
return 0;
}
return em1.getAge() - em2.getAge() > 0 ? -1:1;
}
});
employees.forEach(System.out::println);
~~~
最終的打印結果如下,按照年齡的自定義規則進行排序。
~~~
Employee(id=8, age=79, gender=M, firstName=Alex, lastName=Gussin)
Employee(id=7, age=68, gender=F, firstName=Melissa, lastName=Roy)
Employee(id=10, age=45, gender=M, firstName=Naveen, lastName=Jain)
Employee(id=3, age=43, gender=M, firstName=Ricky, lastName=Martin)
Employee(id=4, age=26, gender=M, firstName=Jon, lastName=Lowman)
Employee(id=1, age=23, gender=M, firstName=Rick, lastName=Beethovan)
Employee(id=5, age=19, gender=F, firstName=Cristine, lastName=Maria)
Employee(id=9, age=15, gender=F, firstName=Neetu, lastName=Singh)
Employee(id=6, age=15, gender=M, firstName=David, lastName=Feezor)
Employee(id=2, age=13, gender=F, firstName=Martina, lastName=Hengis)
~~~
這段代碼如果以lambda表達式簡寫。箭頭左側是參數,右側是函數體,參數類型和返回值根據上下文自動判斷。如下:
~~~
employees.sort((em1,em2) -> {
if(em1.getAge() == em2.getAge()){
return 0;
}
return em1.getAge() - em2.getAge() > 0 ? -1:1;
});
employees.forEach(System.out::println);
~~~
- 前言
- 1.lambda表達式會用了么
- 2.初識Stream-API
- 3.Stream的filter與謂語邏輯
- 4.Stream管道流的map操作
- 5.Stream的狀態與并行操作
- 6.Stream性能差?不要人云亦云
- 7.像使用SQL一樣排序集合
- 8.函數式接口Comparator
- 9.Stream查找與匹配元素
- 10.Stream集合元素歸約
- 11.StreamAPI終端操作
- 12.java8如何排序Map
- Stream流逐行文件處理
- java8-forEach(持續發布中)
- 筆者其它作品推薦
- vue深入淺出系列
- 手摸手教你學Spring Boot2.0
- Spring Security-JWT-OAuth2一本通
- 實戰前后端分離RBAC權限管理系統
- 實戰SpringCloud微服務從青銅到王者