函數式接口是指:有且僅有一個抽象方法的接口。java
函數式接口即適用於函數式編程場景的接口。而Java中函數式編程體現的就是Lambda,因此Lambda接口就是適用於Lambda使用的接口。有且只有一個抽象方法,Java中的Lambda才能順利推導。編程
只要確保接口的有且僅有一個抽象方法便可:數組
修飾符 interface 接口{ public abstract 返回值類型 方法名稱(可選參數信息); //其餘非抽象方法 }
接口中的public abstract能夠省略,因此格式能夠簡化:app
修飾符 interface 接口{ 返回值類型 方法名稱(可選參數信息); //其餘非抽象方法 }
函數式接口能夠做爲方法的參數和返回值類型函數式編程
@FunctionalInterface 修飾符 interface 接口{ 返回值類型 方法名稱(可選參數信息); //其餘非抽象方法 }
檢測該函數是不是函數式接口,是否有且僅有一個抽象方法。函數
public class LambdaTest{ public static void useLambdaInterfaceMethod(LambdaInterface lifm){ lifm.LambdaInterfaceMethod(); } public static void main(String[] args){ //傳入接口的實現類: //useLambdaInterfaceMethod(new 接口的實現類); //傳入接口的匿名內部類 /* useLambdaInterfaceMethod(new LambdaInterface(){ public void LambdaInterfaceMethod(){ System.out.println("這裏傳入的是接口的實現匿名內部類"); } }); */ //抽象方法有且僅有一行語句,能夠省略大括號 //ustLambdaInterfaceMethod(() -> System.out.println("...")); useLambdaInterfaceMethod(() -> { System.out.println("使用Lambda表達式,重寫接口中的抽象方法。") } ); } } //註解FunctionanInterface檢測是否爲函數式接口 @FunctionalInterface interface LambdaInterface{ void LambdaInterfaceMethod(); }
有些場景代碼執行之後,結果不必定使用,從而形成浪費。而Lambda表達式是延遲執行的,能夠做爲解決方案,提高性能。性能
日誌能夠快速定位問題,記錄程序運行中的狀況,以便項目的監控和優化。優化
一種場景是對參數進行有條件的使用,例如在日子消息拼接之後,在知足條件時打印輸出。ui
/* 下面存在性能浪費的問題 調用log方法的時候,傳入第二個參數是一個拼接後的字符串 是先把字符串拼接好,而後再調用log方法 log方法中的日誌等級不是1級,那麼就不會是如此拼接後的字符串 這裏的字符串就白拼接了,形成浪費 */ public class LoggerTest{ public static void log(int level, String msg){ if(level == 1){ System.out.println(msg); } } public static void main(String[] args){ String str1 = "hello"; String str2 = "java"; log(1, str1 + str2); } }
/* 使用Lambda優化日誌: 延遲加載 使用前提:必須存在函數式接口 使用Lambda表達式做爲參數傳遞,僅僅是把參數傳遞到log方法中, 只有知足條件,日誌的等級是1級 纔會調用接口LambdaInterface方法中的msgAdd()方法 而後進行方法的拼接 若是條件不知足,那麼接口LambdaInterface方法中的msgAdd()方法不會執行 拼接字符串的代碼也不會執行,不會存在性能的浪費 */ public class LoggerTest{ public static void log(int level, LambdaInterface li){ if(level == 1){ String str = li.msgAdd(); System.out.println(str); } } public static void main(String[] args){ String str1 = "hello"; String str2 = "java"; log(1, () -> { //返回一個拼接好的字符串 return str1 + str2; }); } } @FunctionalInterface interface LambdaInterface{ //定義一個拼接消息的方法,返回拼接的消息 String msgAdd(); }
public static void main(String[] args){ String str1 = "hello"; String str2 = "java"; log(1, () -> { //沒有知足條件這裏不會執行 System.out.println("不知足條件"); //返回一個拼接好的字符串 return str1 + str2; }); }
Java的Lambda表達式能夠被當作匿名內部類的替代品,若是方法的參數是一個函數式接口類型,那麼就可使用Lambda表達式進行替代。使用Lambda表達式做爲方法參數,其實就是使用函數式接口做爲方法參數。
例如:Runnable接口就是一個函數式接口this
public class Demo{ private static void startThread(Runnable runab){ new Thread(runab).start(); } public static void main(String[] args){ startThread(() -> System.out.println(...)); } }
public class DemoComparator{ //返回值類型是一個函數式接口 //Comparator只有一個抽象方法,因此是函數式接口 private static Comparator<String> newComparator(){ return (a, b) -> b.length() - a.length(); } public static void main(String[] args){ String[] arr = {"a", "abs", "dfae"}; //排序 Arrays.sort(arr, newComparator()); } }
若是返回值類型是一個函數式接口,就能夠直接返回一個Lambda表達式,如上程序。
注:Comparator有兩個抽象方法,int cimpare(T o1, T o2)和boolean equals(Object obj),若是接口聲明瞭一個覆蓋java.lang.Object的全局方法之一的抽象方法,那麼它不會計入接口的抽象方法數量中,由於接口的任何實現都將具備java.lang.Object或其餘地方的實現。
java.util.function.Supplier
import java.util.function.Supplier; public class SupplierClass{ public static String getString(Supplier<String> sup){ return sup.get(); } public static void main(String[] args){ String str1 = "hello"; String str2 = "java"; String getStringMethod = getString(() -> str1 + str2); System.out.println(getStringMethod); } }
Supplier
import java.util.function.Supplier; public class SupplierClass{ //返回的是Integer類型 public static Integer getMax(Supplier<Integer> sup){ return sup.get(); } public static void main(String[] args){ int[] iArr = {12, 252, -12, 435}; int getMaxInteger = getMax(() -> { int max = iArr[0]; //遍歷數組 for(int i : iArr){ //若是數組有一個值大於max,則把這個值賦值給max if(i > max){ max = i; } } //返回數組中的最大值 return max; }); System.out.println(getMaxInteger); } }
java.util.function.Consumer
import java.util.function.Consumer; public class ConsumerClass{ public static void consumerUseMethod(Consumer<String> com){ con.accept("被使用的字符串"); } public static void main(String[] args){ consumerUseMethod((x) -> System.out.println(x)); } }
import java.util.function.Consumer; public class ConsumerClass{ public static void consumerUseMethod(String name,Consumer<String> con){ con.accept(name); } public static void main(String[] args){ consumerUseMethod("lilei", (x) -> System.out.println(x)); } }
若是一個方法的參數和返回值全是Consumer類型,就能夠實現這個效果:消費數據的時候,首先作一個操做,而後再作一個操做,實現組合。而這個方法就是Consumer接口中default方法antThen:
default Consumer<T> andThen(Consumer<? super T> after){ Objects.requireNonNull(after); return (T t) -> { accept(t); after.accept(t); }; }
java.util.Objects和requireNonNull靜態方法將會在參數爲null時主動拋出異常。省去重寫if語句和拋出空指針異常的麻煩。
import java.util.function.Consumer; /* Consumer接口默認方法andThen 做用:須要兩個Consumer接口,能夠把兩個Consumer接口組合到一塊兒,對數據進行消費 例如: Consumer<String> con1 Consumer<String> con2 String s = 「hello」; con1.accept(s); con2.accept(s); 鏈接兩個Consumer接口,再進行消費 con1.andThen(con2).accept(s); */ public class ConsumerClass{ public static void consumerUseMethod(String name, Consumer<String> con, Consumer<String> con2){ //con.accept(name); //con2.accept(name); //使用andThen方法替代上面代碼: con.andThen(con2).accept(name); //con連接的con2,因此先執行con,再執行con2 } public static void main(String[] args){ consumerUseMethod( "lilei", //第一個Lambda表達式 (x) -> System.out.println(x), //第二個Lambda表達式 (x) -> System.out.println("第二個Lambda表達式") ); } }
若是須要多個Consumer接口,使用andThen方法,就是con1.andThen(con2).andThen(con3)...andThen(conN).accept(t);
包含一個抽象方法:boolean test(T t),用於條件判斷場景
符合條件:返回true
不符合:返回false
import java.util.function.Predicate; public class PredicateClass{ public static boolean checkString(String s, Predicate<String> pre){ //傳入要判斷的文本 return pre.test(s); } public static void main(String[] args){ String str = "guess how long"; //判斷字符串長度是否大於5,並返回結果 boolean b = checkString(str, (String s) -> {return str.length() > 5;}); System.out.println(b); } }
省略一些代碼
//省略String s 中類型,能夠直接推導 //只有一行語句,省略大括號和return關鍵字 boolean b = checkString(str, s-> str.length() > 5);
將兩個Predicate條件使用與邏輯鏈接起來實現而且的效果,使用默認方法and
default Predicate<T> and(Predicate<? super T> other){ Objects.requireNonNull(other); return (t) -> test(t) && other.test(t); }
實現:
import java.util.function.Predicate; public class PredicateClass{ public static void checkString(String str, Predicate<String> preOne, Predicate<String> preTwo){ //preOne.test("str")&&preTwo.test("str"); boolean b = preOne.and(preTwo).test(str); System.out.println(b); } public static void main(String[] args){ String strName = "helloworld"; checkString(strName, s-> s.contains("H"),s-> s.contains("W")); } }
default Predicate<T> or(Predicate<? super T> other){ Objects.requireNonNull(Other); return (t) -> test(t) || other.test(t); }
實現:
import java.util.function.Predicate; public class PredicateClass{ public static boolean checkString(String str, Predicate<String> preOne, Predicate<String> preTwo){ //等價於preOne.test(t) || preTwo.test(t); return preOne.or(preTwo).test(str); } public static void main(String[] args){ String strName = "helloworld"; boolean b = checkString(strName, s-> s.contains("h"),s-> s.contains("W")); System.out.println(b); } }
default Predicate<T> negate(){ retuan (t) -> !test(t); }
//取反 return preOne.negate(preTwo).test(str);
用來根據一個類型的數據獲得另外一個類型的數據,前者稱爲前置條件,後者稱爲後置條件。
Function最主要的抽象方法爲:R apply(T t),根據類型T的參數獲取類型R的結果。
import java.util.function.Function; public class FunctionClass{ //Function<原始類型, 要轉換的類型> public static void applyMethod(String str, Function<String, Integer> fun){ //字符串類型轉換成int類型 int num = fun.apply(str); System.out.println(num + 20); } public static void main(String[] args){ String str = "100"; applyMethod(str, s -> Integer.parseInt(s)); } }
//用來組合操做 default <V> Function<T, V> andThen(Function<? super R, ? extends V> after){ Objects.requireNonNull(after); return (T t) -> after.apply(apply(t)); }
實現代碼:
import java.util.function.Function; public class FunctionClass{ //andThen組合操做 public static void applyMethod(String str, Function<String, Integer> funOne, Function<Integer, String> funTwo){ //字符串類型轉換成int類型,加10 //int num = fun.apply(str) + 10; //而後轉換成String類型 //String str = funTwo.apply(num); //組合,先實現funOne,再實現funTwo String strr = funOne.andThen(funTwo).apply(str); System.out.println(strr); } public static void main(String[] args){ String str = "100"; //先把字符串轉換爲integer,結果加10 //把前面funOne的結果轉換成String類型 applyMethod(str, (String s) -> Integer.parseInt(s) + 10, (Integer in) -> String.valueOf(in)); } }
雙冒號::是爲引用運算符,而它所在的表達式被稱爲方法引用。若是Lambda要表達是函數方案已經存在於某個方法的實現中,那麼則能夠經過雙冒號來引用該方法做爲Lambda的替代者。
語義分析
例如:System.out對象中有一個重載的println(String)方法剛好就是咱們所須要的,那麼對於該方法的函數式接口參數,如下兩種方式等價:
Lambda表達式: s -> System.out.println(s); 方法引用寫法: System.out::println
第一種語義是:拿到參數以後經Lambda手,繼而傳遞給System.out.println方法去處理;
第二種是:直接讓System.out中的println方法來取代Lambda。
兩種寫法的執行效果徹底同樣。而第二種方法引用複用了已有方案,更加簡潔。
Lambda中傳遞的參數必定是方法引用中那個方法能夠接收的類型,不然拋出異常。
推導與省略
若是使用Lambda,那麼根據可推導就是可省略的原則,無需指定參數類型,也無需指定重載形式--它們都是自動推導,而若是使用方法引用,也是一樣能夠根據上下文進行推導。
函數式接口是Lambda的基礎,而方法引用是Lambda的孿生兄弟。
/* 經過對象名引用成員方法 使用前提:對象名已經存在,成員方法已經存在 */ public class LambdaObjectUseMethod{ //定義一個方法 public static void printString(Printable p){ p.print("hello"); } public static void main(String[] args){ //Lambda表達式 printString(x -> { MethodUpper mu = new MethodUpper(); mu.printUpperCaseString(x); }); //方法引用 //對象必須已經存在 MethodUpper mu2 = new MethodUpper(); printString(mu2::printUpperCaseString); } } //函數式接口 @FunctionalInterface interface Printable{ void print(String s); } //將字符串全改成大寫並輸出 class MethodUpper{ public void printUpperCaseString(String str){ System.out.println(str.toUpperCase()); } }
/* 經過對象名引用成員方法 使用前提:類已經存在,靜態方法已經存在 */ public class LambdaObjectUseMethod{ //定義一個方法 public static void printString(Printable p){ p.print("hello"); } public static void main(String[] args){ //經過類名引用靜態方法 printString(MethodUpper::printUpperCaseString); } } //函數式接口 @FunctionalInterface interface Printable{ void print(String s); } //將字符串全改成大寫並輸出 class MethodUpper{ //靜態方法 public static void printUpperCaseString(String str){ System.out.println(str.toUpperCase()); } }
//函數式接口 @FunctionalInterface interface Greetable{ void greet(); } //父類 class FatherClass{ public void showMethod(){ System.out.println("我是你爸爸"); } } public class SonClass extends FatherClass{ public void showMethod(){ System.out.println("不符solo!!!"); } public void useGreetable(Greetable gt){ gt.greet(); } public void show(){ /* useGreetable(() -> { FatherClass fc = new SonClass(); super.showMethod(); }); */ //super引用成員方法 useGreetable(super::showMethod); } public static void main(String[] args){ new SonClass().show(); } }
//函數式接口 @FunctionalInterface interface Greetable{ void greet(); } //父類 class FatherClass{ public void showMethod(){ System.out.println("我是你爸爸"); } } public class SonClass extends FatherClass{ public void showMethod(){ System.out.println("不符solo!!!"); } public void useGreetable(Greetable gt){ gt.greet(); } public void show(){ //this引用本類成員方法 useGreetable(this::showMethod); } public static void main(String[] args){ new SonClass().show(); } }
構造器引用使用 類名稱::new 的格式
//類構造器引用 class Person{ String name; public Person(String name){ this.name = name; } public String getName(){ return name; } public void setName(String name){ this.name = name; } } @FunctionalInterface interface CreatePerson{ Person returnPerson(String name); } public class PersonFunctionClass{ public static void method(String name, CreatePerson cp){ Person p = cp.returnPerson(name); //Person p = System.out.println(p.getName()); } public static void main(String[] args){ /*Lambda表達式 method("lilei",(String name) -> { return new Person(name); }); */ /*簡化版的Lambda表達式 method("lilei", name -> new Person(name)); */ //構造器引用 method("lilei", Person::new); } }
//數組的構造器引用 @FunctionalInterface interface ArrayBuiler{ //建立int數組類型的方法 int[] createIntArray(int length); } //數組的構造器引用 public class ArrayTestClass{ //定義一個方法,方法參數傳遞建立數組的長度和ArrayBuiler接口 //方法內部根據傳遞的長度建立數組 public static int[] createArray(int length, ArrayBuiler ab){ return ab.createIntArray(length); } public static void main(String[] args){ int[] arr1 = createArray(10, (int i) -> { return new int[i]; }); int[] arr2 = createArray(12, len -> new int[len]); //引用數組的構造器 int[] arr3 = createArray(13, int[]::new); } }