1.Lambda簡介
“Lambda 表達(dá)式”(lambdaexpression)是一個(gè)匿名函數(shù)晦闰,Lambda表達(dá)式基于數(shù)學(xué)中的λ演算得名允蜈。我們可以把Lambda表達(dá)式理解為簡潔地可傳遞的匿名函數(shù)的一種方式:它沒有名稱,但它有參數(shù)列表选酗,函數(shù)主體阵难,返回類型,可能還有一個(gè)可以拋出的異常列表芒填。
- 匿名——匿名這里的意思是無需寫函數(shù)名字也就是方法名稱呜叫。
- 函數(shù)——它也是一個(gè)函數(shù)空繁,因?yàn)長ambda函數(shù)不像方法那樣屬于特定的類。但和方法一樣朱庆,都會有參數(shù)列表盛泡,函數(shù)主體,返回類型娱颊,還可能有可以跑出的異常傲诵。
- 傳遞——以前java不支持傳遞方法,現(xiàn)在有了Lambda表達(dá)式以后可以把其作為參數(shù)傳遞給方法维蒙。
- 簡介——無需像匿名類那樣寫很多模板代碼掰吕。
舉個(gè)例子:
JAVA7:
Comparator<Apple> byWeight = new Comparator<Apple>() {
public int compare(Apple a1, Apple a2){
return a1.getWeight().compareTo(a2.getWeight());
}
};
JAVA8:
Comparator<Apple> byWeight =(Apple a1, Apple a2) -> a1.getWeight().compareTo(a2.getWeight());
上面描述兩個(gè)蘋果比較重量的例子,這里我們可以看出我們的Lambda表達(dá)式的代碼更加小并且精巧颅痊。
2.Lambda講解
對于我們第一節(jié)中的Lambda表達(dá)式有三個(gè)部分:
- 參數(shù)列表 (Apple a1, Apple a2)
- 箭頭 -> 用于把參數(shù)列表和Lambda分離開
- Lambda主體 a1.getWeight().compareTo(a2.getWeight())
Java語言設(shè)計(jì)者選擇這樣的語法,是因?yàn)镃#和Scala等語言中的類似功能廣受歡迎局待。 Lambda的基本語法是:
(parameters) -> expression 自帶return斑响,后面跟表達(dá)式
(parameters) -> { statements; } 不帶return 需要手動設(shè)置,后面跟語句
2.1哪里使用Lambda表達(dá)式钳榨?
2.1.1函數(shù)式接口
一言以蔽之舰罚,函數(shù)式接口就是只定義一個(gè)抽象方法的接口。舉幾個(gè)例子:
public interface Comparator<T> {
int compare(T o1, T o2);
}
public interface Runnable{
void run();
}
public interface ActionListener extends EventListener{
void actionPerformed(ActionEvent e);
}
public interface Callable<V>{
V call();
}
public interface PrivilegedAction<V>{
V run();
}
上面的都是函數(shù)接口薛耻,因?yàn)樗麄冊谧约旱慕涌谥兄挥幸粋€(gè)方法营罢。對于函數(shù)式接口,Lambda表達(dá)式允許我們直接用內(nèi)聯(lián)的方式為函數(shù)式接口的抽象方法提供實(shí)現(xiàn)饼齿,并把整個(gè)表達(dá)式作為函數(shù)式接口的實(shí)例饲漾。你用匿名內(nèi)部類也可以完成同樣的事情,匿名內(nèi)部類會有點(diǎn)笨重缕溉,代碼量會增加考传。
Runnable r1 = () -> System.out.println("Hello World 1"); //lambda
Runnable r2 = new Runnable(){
public void run(){
System.out.println("Hello World 2");
}
}; //匿名內(nèi)部類
public static void process(Runnable r){
r.run();
}
process(r1);
process(r2);
process(() -> System.out.println("Hello World 3")); //lambda
在java8中引入了一個(gè)注解@FunctionalInterface。這個(gè)注解用于表示該接口會設(shè)計(jì)成一個(gè)函數(shù)式接口证鸥。如果你用@FunctionalInterface定義了一個(gè)接口僚楞,而它卻不是函數(shù)式接口的話,編譯器將返回一個(gè)提示原因的錯(cuò)誤枉层。@FunctionalInterface不是必須的但是為了良好的接口設(shè)計(jì)就像@Override一樣,用來表示方法為重寫了泉褐。
3.函數(shù)式接口詳解
如上面所說,函數(shù)式接口值定義了一個(gè)方法鸟蜡。函數(shù)式接口的抽象方法的簽名稱為函數(shù)描述符膜赃。所以為了應(yīng)用不同的Lambda表達(dá)式,java8也為我們提供了一些常見函數(shù)描述符的函數(shù)式接口矩欠。在我們的java.util.function包中有幾個(gè)新引入的需要介紹财剖。
3.1Predicate
java.util.function.Predicate<T>接口定義了一個(gè)名叫test的抽象方法悠夯,它接受泛型T對象,并返回一個(gè)boolean躺坟。
使用代碼如下:
@FunctionalInterface
public interface Predicate<T>{
boolean test(T t);
}
public static <T> List<T> filter(List<T> list, Predicate<T> p) {
List<T> results = new ArrayList<>();
for(T s: list){
if(p.test(s)){
results.add(s);
}
}
return results;
}
Predicate<String> nonEmptyStringPredicate = (String s) -> !s.isEmpty();
List<String> nonEmpty = filter(listOfStrings, nonEmptyStringPredicate);
在Predicate<T>中有幾個(gè)default方法用來加強(qiáng)其使用沦补,and,or等等。
3.2Consumer
Consumer顧名思義消費(fèi)咪橙,只消費(fèi)參數(shù)不返回參數(shù)夕膀。它定義了一個(gè)accept方法,接受泛型T的對象美侦,沒有返回产舞。你如果需要訪問類型T的對象,并對其執(zhí)行某些操作菠剩,就可以使用這個(gè)接口易猫。
@FunctionalInterface
public interface Consumer<T>{
void accept(T t);
}
public static <T> void forEach(List<T> list, Consumer<T> c){
for(T i: list){
c.accept(i);
}
}
forEach(
Arrays.asList(1,2,3,4,5),
(Integer i) -> System.out.println(i)
);
上面展示了如何打印list中的每個(gè)元素,在stream中也會有foreach傳遞的也是我們的Consumer。
3.3Function
java.util.function.Function<T,R>接口定義了一個(gè)叫作apply的方法具壮,它接受一個(gè)泛型T的對象准颓,并返回一個(gè)泛型R的對象。如果你需要定義一個(gè)Lambda棺妓,將輸入對象的信息映射到輸出攘已,就可以使用這個(gè)接口。同樣的在stream中有一個(gè)map方法參數(shù)也是這個(gè)怜跑,下面我們將模仿map方法样勃。
public static <T,R> List<R> map(List<T> list, Function<T,R> function){
List<R> result = new ArrayList<R>();
for(T s : list){
result.add(function.apply(s));
}
return result;
}
List<String> list = map(Arrays.asList(1,2,3),(Integer i)-> String.valueOf(i));
上面展現(xiàn)了如何把一個(gè)Integer類型的字符列表轉(zhuǎn)換成String類型的字符列表。
3.4基本類型的設(shè)計(jì)
java8為了讓我們對基本類型更好的使用性芬,提供了基本類型的函數(shù)接口峡眶,主要是為了節(jié)約我們的內(nèi)存,如果我們自動裝箱拆箱批旺,會占用更多內(nèi)存幌陕。所以提供了一些DoublePredicate、 IntConsumer汽煮、 LongBinaryOperator搏熄、 IntFunction等的函數(shù)式接口讓我們使用。
4.lambda綜合實(shí)戰(zhàn)
下面我們會對Apple對象進(jìn)行排序暇赤,我們會按照重量的策略進(jìn)行排序
4.1最簡單的
在java的集合中提供了一個(gè)排序的方法心例,sort方法,方法簽名如下:
void sort(Comparator<? super E> c)
第一個(gè)方案也是我們最簡單的:
public static class AppleComparator implements Comparator<Apple>{
@Override
public int compare(Apple o1, Apple o2) {
return o1.getWeight().compareTo(o2.getWeight());
}
}
inventory.sort(new AppleComparator());
4.2匿名內(nèi)部類
inventory.sort(new Comparator<Apple>() {
public int compare(Apple a1, Apple a2){
return a1.getWeight().compareTo(a2.getWeight());
}
});
4.3使用Lambda表達(dá)式
Comparator是函數(shù)式接口我們可以使用lambda表達(dá)式來改造
inventory.sort((Apple a1, Apple a2)-> a1.getWeight().compareTo(a2.getWeight()));
這里我們可以不用加屬性類型鞋囊,更加簡單:
inventory.sort((a1, a2) -> a1.getWeight().compareTo(a2.getWeight()));
當(dāng)然在我們還能更簡單在Comparator中有一個(gè)comparing方法的靜態(tài)輔助方法止后,它可以接受一個(gè)Function。
inventory.sort(Comparator.comparing(a -> a.getWeight()));
4.4方法引用
使用lambda的語法糖可以更加簡單
inventory.sort(Comparator.comparing(Apple::getWeight));