前言
成都創(chuàng)新互聯(lián)專業(yè)為企業(yè)提供色尼網(wǎng)站建設(shè)、色尼做網(wǎng)站、色尼網(wǎng)站設(shè)計(jì)、色尼網(wǎng)站制作等企業(yè)網(wǎng)站建設(shè)、網(wǎng)頁(yè)設(shè)計(jì)與制作、色尼企業(yè)網(wǎng)站模板建站服務(wù),10余年色尼做網(wǎng)站經(jīng)驗(yàn),不只是建網(wǎng)站,更提供有價(jià)值的思路和整體網(wǎng)絡(luò)服務(wù)。
最近在學(xué)習(xí)java8,所以接下來(lái)會(huì)給大家介紹一系列的Java8學(xué)習(xí)內(nèi)容,那么讓我們先從lambda表達(dá)式開(kāi)始。
眾所周知從java8出現(xiàn)以來(lái)lambda是最重要的特性之一,它可以讓我們用簡(jiǎn)潔流暢的代碼完成一個(gè)功能。 很長(zhǎng)一段時(shí)間java被吐槽是冗余和缺乏函數(shù)式編程能力的語(yǔ)言,隨著函數(shù)式編程的流行java8種也引入了 這種編程風(fēng)格。在此之前我們都在寫(xiě)匿名內(nèi)部類干這些事,但有時(shí)候這不是好的做法,本文中將介紹和使用lambda, 帶你體驗(yàn)函數(shù)式編程的魔力。
什么是lambda?
lambda表達(dá)式是一段可以傳遞的代碼,它的核心思想是將面向?qū)ο笾械膫鬟f數(shù)據(jù)變成傳遞行為。 我們回顧一下在使用java8之前要做的事,之前我們編寫(xiě)一個(gè)線程時(shí)是這樣的:
Runnable r = new Runnable() { @Override public void run() { System.out.println("do something."); } }
也有人會(huì)寫(xiě)一個(gè)類去實(shí)現(xiàn)Runnable接口,這樣做沒(méi)有問(wèn)題,我們注意這個(gè)接口中只有一個(gè)run方法, 當(dāng)把Runnable對(duì)象給Thread對(duì)象作為構(gòu)造參數(shù)時(shí)創(chuàng)建一個(gè)線程,運(yùn)行后將輸出do something.
。 我們使用匿名內(nèi)部類的方式實(shí)現(xiàn)了該方法。
這實(shí)際上是一個(gè)代碼即數(shù)據(jù)的例子,在run方法中是線程要執(zhí)行的一個(gè)任務(wù),但上面的代碼中任務(wù)內(nèi)容已經(jīng)被規(guī)定死了。 當(dāng)我們有多個(gè)不同的任務(wù)時(shí),需要重復(fù)編寫(xiě)如上代碼。
設(shè)計(jì)匿名內(nèi)部類的目的,就是為了方便 Java 程序員將代碼作為數(shù)據(jù)傳遞。不過(guò),匿名內(nèi)部 類還是不夠簡(jiǎn)便。 為了執(zhí)行一個(gè)簡(jiǎn)單的任務(wù)邏輯,不得不加上 6 行冗繁的樣板代碼。那如果是lambda該怎么做?
Runnable r = () -> System.out.println("do something.");
嗯,這代碼看起來(lái)很酷,你可以看到我們用()和->的方式完成了這件事,這是一個(gè)沒(méi)有名字的函數(shù),也沒(méi)有人和參數(shù),再簡(jiǎn)單不過(guò)了。 使用->將參數(shù)和實(shí)現(xiàn)邏輯分離,當(dāng)運(yùn)行這個(gè)線程的時(shí)候執(zhí)行的是->之后的代碼片段,且編譯器幫助我們做了類型推導(dǎo); 這個(gè)代碼片段可以是用{}包含的一段邏輯。下面一起來(lái)學(xué)習(xí)一下lambda的語(yǔ)法。
基礎(chǔ)語(yǔ)法
在lambda中我們遵循如下的表達(dá)式來(lái)編寫(xiě):
expression = (variable) -> action
可以看到Java中l(wèi)ambda表達(dá)式的格式:參數(shù)、箭頭、以及動(dòng)作實(shí)現(xiàn),當(dāng)一個(gè)動(dòng)作實(shí)現(xiàn)無(wú)法用一行代碼完成,可以編寫(xiě) 一段代碼用{}包裹起來(lái)。
lambda表達(dá)式可以包含多個(gè)參數(shù),例如:
int sum = (x, y) -> x + y;
這時(shí)候我們應(yīng)該思考這段代碼不是之前的x和y數(shù)字相加,而是創(chuàng)建了一個(gè)函數(shù),用來(lái)計(jì)算兩個(gè)操作數(shù)的和。 后面用int類型進(jìn)行接收,在lambda中為我們省略去了return。
函數(shù)式接口
函數(shù)式接口是只有一個(gè)方法的接口,用作lambda表達(dá)式的類型。前面寫(xiě)的例子就是一個(gè)函數(shù)式接口,來(lái)看看jdk中的Runnable源碼
@FunctionalInterface public interface Runnable { /** * When an object implementing interface <code>Runnable</code> is used * to create a thread, starting the thread causes the object's * <code>run</code> method to be called in that separately executing * thread. * <p> * The general contract of the method <code>run</code> is that it may * take any action whatsoever. * * @see java.lang.Thread#run() */ public abstract void run(); }
這里只有一個(gè)抽象方法run,實(shí)際上你不寫(xiě)public abstract
也是可以的,在接口中定義的方法都是public abstract的。 同時(shí)也使用注解@FunctionalInterface
告訴編譯器這是一個(gè)函數(shù)式接口,當(dāng)然你不這么寫(xiě)也可以,標(biāo)識(shí)后明確了這個(gè)函數(shù)中 只有一個(gè)抽象方法,當(dāng)你嘗試在接口中編寫(xiě)多個(gè)方法的時(shí)候編譯器將不允許這么干。
嘗試函數(shù)式接口
我們來(lái)編寫(xiě)一個(gè)函數(shù)式接口,輸入一個(gè)年齡,判斷這個(gè)人是否是成人。
public class FunctionInterfaceDemo { @FunctionalInterface interface Predicate<T> { boolean test(T t); } /** * 執(zhí)行Predicate判斷 * * @param age 年齡 * @param predicate Predicate函數(shù)式接口 * @return 返回布爾類型結(jié)果 */ public static boolean doPredicate(int age, Predicate<Integer> predicate) { return predicate.test(age); } public static void main(String[] args) { boolean isAdult = doPredicate(20, x -> x >= 18); System.out.println(isAdult); } }
從這個(gè)例子我們很輕松的完成 是否是成人 的動(dòng)作,其次判斷是否是成人,在此之前我們的做法一般是編寫(xiě)一個(gè) 判斷是否是成人的方法,是無(wú)法將 判斷 共用的。而在本例只,你要做的是將 行為 (判斷是否是成人,或者是判斷是否大于30歲) 傳遞進(jìn)去,函數(shù)式接口告訴你結(jié)果是什么。
實(shí)際上諸如上述例子中的接口,偉大的jdk設(shè)計(jì)者為我們準(zhǔn)備了java.util.function包
我們前面寫(xiě)的Predicate函數(shù)式接口也是JDK種的一個(gè)實(shí)現(xiàn),他們大致分為以下幾類:
接口 | 參數(shù) | 返回值 | 類別 | 示例 |
---|---|---|---|---|
Consumer | T | void | 消費(fèi)型接口 | 輸出一個(gè)值 |
Supplier | None | T | 供給型接口 | 工廠方法 |
Function | T | R | 函數(shù)型接口 | 獲得 Artist 對(duì)象的名字 |
Predicate | T | boolean | 斷言型接口 | 這張唱片已經(jīng)發(fā)行了嗎 |
消費(fèi)型接口示例
public static void donation(Integer money, Consumer<Integer> consumer){ consumer.accept(money); } public static void main(String[] args) { donation(1000, money -> System.out.println("好心的麥樂(lè)迪為Blade捐贈(zèng)了"+money+"元")) ; }
供給型接口示例
public static List<Integer> supply(Integer num, Supplier<Integer> supplier){ List<Integer> resultList = new ArrayList<Integer>() ; for(int x=0;x<num;x++) resultList.add(supplier.get()); return resultList ; } public static void main(String[] args) { List<Integer> list = supply(10,() -> (int)(Math.random()*100)); list.forEach(System.out::println); }
函數(shù)型接口示例
轉(zhuǎn)換字符串為Integer
public static Integer convert(String str, Function<String, Integer> function) { return function.apply(str); } public static void main(String[] args) { Integer value = convert("28", x -> Integer.parseInt(x)); }
斷言型接口示例
篩選出只有2個(gè)字的水果
public static List<String> filter(List<String> fruit, Predicate<String> predicate){ List<String> f = new ArrayList<>(); for (String s : fruit) { if(predicate.test(s)){ f.add(s); } } return f; } public static void main(String[] args) { List<String> fruit = Arrays.asList("香蕉", "哈密瓜", "榴蓮", "火龍果", "水蜜桃"); List<String> newFruit = filter(fruit, (f) -> f.length() == 2); System.out.println(newFruit); }
默認(rèn)方法
在Java語(yǔ)言中,一個(gè)接口中定義的方法必須由實(shí)現(xiàn)類提供實(shí)現(xiàn)。但是當(dāng)接口中加入新的API時(shí), 實(shí)現(xiàn)類按照約定也要修改實(shí)現(xiàn),而Java8的API對(duì)現(xiàn)有接口也添加了很多方法,比如List接口中添加了sort方法。 如果按照之前的做法,那么所有的實(shí)現(xiàn)類都要實(shí)現(xiàn)sort方法,JDK的編寫(xiě)者們一定非常抓狂。
幸運(yùn)的是我們使用了Java8,這一問(wèn)題將得到很好的解決,在Java8種引入新的機(jī)制,支持在接口中聲明方法同時(shí)提供實(shí)現(xiàn)。 這令人激動(dòng)不已,你有兩種方式完成 1.在接口內(nèi)聲明靜態(tài)方法 2.指定一個(gè)默認(rèn)方法。
我們來(lái)看看在JDK8中上述List接口添加方法的問(wèn)題是如何解決的
default void sort(Comparator<? super E> c) { Object[] a = this.toArray(); Arrays.sort(a, (Comparator) c); ListIterator<E> i = this.listIterator(); for (Object e : a) { i.next(); i.set((E) e); } }
翻閱List接口的源碼,其中加入一個(gè)默認(rèn)方法default void sort(Comparator<? super E> c)
。 在返回值之前加入default關(guān)鍵字,有了這個(gè)方法我們可以直接調(diào)用sort方法進(jìn)行排序。
List<Integer> list = Arrays.asList(2, 7, 3, 1, 8, 6, 4); list.sort(Comparator.naturalOrder()); System.out.println(list);
Comparator.naturalOrder()
是一個(gè)自然排序的實(shí)現(xiàn),這里可以自定義排序方案。 你經(jīng)??吹绞褂肑ava8操作集合的時(shí)候可以直接foreach的原因也是在Iterable接口中也新增了一個(gè)默認(rèn)方法:forEach , 該方法功能和 for 循環(huán)類似,但是允許 用戶使用一個(gè)Lambda表達(dá)式作為循環(huán)體。
在后面的章節(jié)中我們?cè)俅瓮ㄟ^(guò)案例來(lái)展示函數(shù)式編程的魅力 :)
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)創(chuàng)新互聯(lián)的支持。
當(dāng)前標(biāo)題:Java8深入學(xué)習(xí)系列(一)lambda表達(dá)式介紹
分享鏈接:http://www.rwnh.cn/article4/gsphoe.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站營(yíng)銷、品牌網(wǎng)站建設(shè)、網(wǎng)站建設(shè)、網(wǎng)站設(shè)計(jì)公司、定制網(wǎng)站、定制開(kāi)發(fā)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)