Lambda深入理解

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));
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市译株,隨后出現(xiàn)的幾起案子瓜喇,更是在濱河造成了極大的恐慌,老刑警劉巖歉糜,帶你破解...
    沈念sama閱讀 212,718評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件乘寒,死亡現(xiàn)場離奇詭異,居然都是意外死亡匪补,警方通過查閱死者的電腦和手機(jī)伞辛,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,683評論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來夯缺,“玉大人蚤氏,你說我怎么就攤上這事∮欢担” “怎么了竿滨?”我有些...
    開封第一講書人閱讀 158,207評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長捏境。 經(jīng)常有香客問我姐呐,道長,這世上最難降的妖魔是什么典蝌? 我笑而不...
    開封第一講書人閱讀 56,755評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮头谜,結(jié)果婚禮上骏掀,老公的妹妹穿的比我還像新娘。我一直安慰自己柱告,他們只是感情好截驮,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,862評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著际度,像睡著了一般葵袭。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上乖菱,一...
    開封第一講書人閱讀 50,050評論 1 291
  • 那天坡锡,我揣著相機(jī)與錄音,去河邊找鬼窒所。 笑死鹉勒,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的吵取。 我是一名探鬼主播禽额,決...
    沈念sama閱讀 39,136評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了脯倒?” 一聲冷哼從身側(cè)響起实辑,我...
    開封第一講書人閱讀 37,882評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎藻丢,沒想到半個(gè)月后剪撬,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,330評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡郁岩,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,651評論 2 327
  • 正文 我和宋清朗相戀三年婿奔,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片问慎。...
    茶點(diǎn)故事閱讀 38,789評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡萍摊,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出如叼,到底是詐尸還是另有隱情冰木,我是刑警寧澤,帶...
    沈念sama閱讀 34,477評論 4 333
  • 正文 年R本政府宣布笼恰,位于F島的核電站踊沸,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏社证。R本人自食惡果不足惜逼龟,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,135評論 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望追葡。 院中可真熱鬧腺律,春花似錦、人聲如沸宜肉。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,864評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽谬返。三九已至之斯,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間遣铝,已是汗流浹背佑刷。 一陣腳步聲響...
    開封第一講書人閱讀 32,099評論 1 267
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留翰蠢,地道東北人项乒。 一個(gè)月前我還...
    沈念sama閱讀 46,598評論 2 362
  • 正文 我出身青樓,卻偏偏與公主長得像梁沧,于是被迫代替她去往敵國和親檀何。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,697評論 2 351

推薦閱讀更多精彩內(nèi)容