摘要:很多語言等從設計之初就支持表達式。注意此時外部局部變量將自動變為作為方法返回值例子返回判斷字符串是否為空判斷字符串是否為空今天關于新特性表達式就講到這里了,接下來我會繼續講述新特性之函數式接口。
上一篇文章我們了解了Java8新特性-接口默認方法,接下來我們聊一聊Java8新特性之Lambda表達式。
Lambda表達式(也稱為閉包),它允許我們將函數當成參數傳遞給某個方法,或者把代碼本身當作數據處理。很多語言(Groovy、Scala等)從設計之初就支持Lambda表達式。但是java中使用的是 匿名內部類代替。
最后借助強大的社區力量,找了一個折中的Lambda實現方案,可以實現簡潔而緊湊的語言結構。
1、匿名內部類到Lambda的演化匿名內部類,即一個沒有名字的,存在于一個類或方法內部的類。當我們需要用某個類且只需要用一次,創建和使用和二為一時,我們可以選擇匿名內部類,省掉我們定義類的步驟。
匿名內部類會隱士的繼承一個類或實現一個接口,或者說匿名內部類是一個繼承了該類或者實現了該接口的子類匿名對象。下面看一個匿名內部類的例子:
測試類中調用方法
package com.lotbyte.main; /* 定義和使用匿名內部類 */ public class NoNameClass { public static void main(String[] args) { Model m = new Model(){ @Override public void func() { System.out.println("方法的實現"); } }; m.func(); } } // 需要被實現的接口 interface Model{ void func(); }2、Lambda快速使用
從某種意義上來說,Lambda表達式可以看作是匿名內部類對象的簡寫形式。最簡單的Lambda表達式可以由 用逗號分隔的參數列表、->符號和語句塊組成。
注意:此時匿名內部類只能實現接口,不能是繼承抽象類.
例如將上面的例子做一個簡化,使用Lambda的形式如下:
public class NonameClassForLambda { public static void main(String[] args) { // Lambda方式簡寫,方法實現可以很簡單 Model1 md = ()-> System.out.println("hello"); md.func(); // 也可以是比較復雜的操作 md = () -> { for (int i = 1; i <=5; i++) { System.out.println(i); } }; md.func(); } } // 接口 interface Model1{ void func(); }
以上是一個簡單的Lambda的書寫形式,()中是形參列表,沒有則為空括號, ->為語法格式,之后則為方法的實現(一條語句可以直接書寫,當有多條語句時,需要使用{}進行包裹)。從這可以看出在接口中必須只能存在一個抽象方法。
注意:Lambda中必須有個接口
3、Lambda的形式使用Lambda時,實現方法可以有參數,也可以有返回值,如果沒指定參數類型,則由編譯器自行推斷得出。
3.1、 無參帶返回值生成[1,10]之間的任意整數
interface Model2{ int func(); } Model2 md2 = () -> {return (int)(Math.random()*10+1)};
說明:Lambda的改寫需要有對應的抽象方法,當沒有參數時需要使用()占位,當表達式只有一行代碼時,可以省略return和{}
以上的Lambda等價于:
Model2 md2 = () -> (int)(Math.random()*10+1);3.2 、帶參帶返回值
返回一個對數字描述的字符串:
interface Model3{ String func(int a); } Model3 md3 = (int a) -> { return "This is a number " + a; };
說明:形參寫在()內即可,參數的類型可以省略,此時將由編譯器自行推斷得出,同時還可以省略()
md3 = a -> "This is a number " + a;
省略了參數類型,小括號,同時連帶實現體的括號和return都省了。
3.3 、帶多個參數根據輸入的運算符計算兩個數的運算,并返回結果:
interface Model4{ String func(int a, int b, String oper); } Model4 md4 = (a, b, s) -> { String res = ""; if("+".equals(s)){ res = ( a+b ) + ""; }else if("-".equals(s)){ res = ( a-b ) + ""; }else if("*".equals(s)){ res = ( a*b ) + ""; }else if("/".equals(s)){ res = ( a/b ) + ""; // 暫不考慮除0的情況 }else{ res = "操作有失誤"; } return res; }; System.out.println(md4.func(1,1,"+"));
以上例子為多個參數的Lambda表達式,其中省略掉了每一個參數的類型,編譯器自動推斷。多條語句時實現體的{}不能省。
4、Lambda作為參數在jdk8之前,接口可以作為方法參數傳入,執行時必須提供接口實現類的實例。從java8開始,Lambda可以作為接口方法實現,當作參數傳入,無論從形式上還是實際上都省去了對象的創建。使代碼更加的緊湊簡單高效。
使用Lambda表達式需要有以下幾步:
? 1、定義接口,抽象方法的模板;
? 2、在某方法中需要接口作為參數;
? 3、調用方法時需要將抽象方法實現(此時我們使用Lambda表達式)并傳入即可。
在接口中,必須有且僅有一個抽象方法,以確定Lambda模板
// 無參無返回值的方法 interface LambdaInterface1{ void printString(); } // 帶參無返回值的方法 interface LambdaInterface2{ void printString(String str); } 4.2、定義方法接收參數 在某方法中需要使用接口作為參數 // 無參 public static void testLambda(LambdaInterface1 lam1){ lam1.printString(); } ? // 帶參 public static void testLambda2(String s,LambdaInterface2 lam2){ lam2.printString(s); }4.3、Lambda實現
使用方法時需要用Lambda將抽象方法實現
使用方法時需要用Lambda將抽象方法實現
// 無參Lambda作為參數 testLambda(()->{ System.out.println("可以簡單,可以復雜"); }); // 帶參Lambda作為參數 testLambdaParam("hello",(a)->{ System.out.println(a); });
通過以上三步,能夠完整地展示Lambda從和演變而來。此后在使用時,jdk中已經提供很多場景了,即前兩部已經完成,我們更多的是實現第三步即可。
5、forEach展示Lambda例如以ArrayList的遍歷為例子,分析Lambda的使用方式。
public static void main(String[] args) { Liststrs = new ArrayList (){ { add("aaa"); add("bbb"); add("ccc"); } }; strs.forEach((str)-> System.out.println(str)); }
下面看看forEach的源碼,定義中使用了接口Consumer作為參數,并調用了其方法:
Consumer中的抽象方法只有accept一個
通過在forEach方法中調用Consumer的accept方法,并將每一個元素作為參數傳入,使得accept方法可以對每一個元素進行操作,當我們使用Lambda實現accept時就變成了我們自己對每一個元素的處理了。我們只負責處理即可。
6、Lambda中使用變量在Lambda中可以定義自己的局部變量,也可以使用外層方法的局部變量,還可以使用屬性。這一點也不難理解,既然是一個方法的實現,只寫了一個代碼塊,那么使用本身所屬方法的局部變量和類的屬性也并不過分。
public static void main(String[] args) { Liststrs = new ArrayList (){ { add("aaa"); add("bbb"); add("ccc"); } }; int j = 1; strs.forEach((str)->{ int i = 0; System.out.println(str + " " + i + " " + j); }); }
注意:此時外部局部變量將自動變為final
7、Lambda作為方法返回值例子:返回判斷字符串是否為空
public class Demo004_2 { public static void main(String[] args) { System.out.println(testLambda().isEmpty("string")); } ? // 判斷字符串是否為空 public static AssertEmpty testLambda(){ return (n)-> null==n||n.trim().isEmpty(n); } } ? interface AssertEmpty{ boolean isEmpty(String str); }
今天關于Java8新特性-Lambda表達式就講到這里了,接下來我會繼續講述Java8新特性之函數式接口。敬請繼續關注!歡迎留言評論。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/74045.html
摘要:上一篇小樂介紹了新特性函數式接口,大家可以點擊回顧。中引入方法引用新特性用于簡化應用對象方法的調用,方法引用是用來直接訪問類或者實例的已經存在的方法或者構造方法。方法引用是一種更簡潔易懂的表達式。 上一篇小樂介紹了《Java8新特性-函數式接口》,大家可以點擊回顧。這篇文章將接著介紹Java8新特性之方法引用。 Java8 中引入方法引用新特性,用于簡化應用對象方法的調用,?方法引用是...
摘要:大家好,我是樂字節的小樂,上一次我們說到了核心特性之函數式接口,接下來我們繼續了解又一核心特性方法引用。方法引用是一種更簡潔易懂的表達式。感謝光臨閱讀小樂的,敬請關注樂字節后續將繼續講述等前沿知識技術。 大家好,我是樂字節的小樂,上一次我們說到了Java8核心特性之函數式接口,接下來我們繼續了解Java8又一核心特性——方法引用。 showImg(https://segmentfaul...
摘要:上一篇小樂帶大家學過新特性表達式,什么時候可以使用通常表達式是用在函數式接口上使用的。使用實現創建產生一個工廠對象以上就是小樂帶給大家的新特性之函數式接口,下一篇將會為大家帶來新特性之方法引用,敬請關注。 上一篇小樂帶大家學過 Java8新特性-Lambda表達式,什么時候可以使用Lambda?通常Lambda表達式是用在函數式接口上使用的。從Java8開始引入了函數式接口,其說明比較...
摘要:大家好,上一篇小樂給大家講述了樂字節核心特性表達式,點擊回顧。接下來繼續核心特性之函數式接口。感謝大家欣賞小樂帶來的核心特性之函數式接口,接下來還會更多核心技術講解,請關注樂字節如需要視頻課程,請搜索樂字節騰訊課堂 大家好,上一篇小樂給大家講述了《樂字節-Java8核心特性-Lambda表達式》,點擊回顧。接下來繼續:Java8核心特性之函數式接口。 什么時候可以使用Lambda?通常...
摘要:使用表達式,使得應用變得簡潔而緊湊。很多語言等從設計之初就支持表達式。表達式的參數與函數式接口內方法的參數,返回值類型相互對應。更多教程和資料請上騰訊課堂樂字節 showImg(https://segmentfault.com/img/bVbtotg?w=935&h=345); Java8 引入Lambda表達式,允許開發者將函數當成參數傳遞給某個方法,或者把代碼本身當作數據進行處理。...
閱讀 2233·2021-09-23 11:52
閱讀 1899·2021-09-02 15:41
閱讀 3018·2019-08-30 10:47
閱讀 1984·2019-08-29 17:14
閱讀 2334·2019-08-29 16:16
閱讀 3192·2019-08-28 18:29
閱讀 3418·2019-08-26 13:30
閱讀 2610·2019-08-26 10:49