函數(shù)式接口
1.1 函數(shù)式接口概述
函數(shù)式接口:有且僅有一個(gè)抽象方法的接口
Java中的函數(shù)式編程體現(xiàn)就是Lambda表達(dá)式,所以函數(shù)式接口就是可以使用與Lambda使用的接口
只有確保接口只能夠有且只有一個(gè)抽象方法,Lambda才能順利的進(jìn)行推導(dǎo)
檢測(cè)接口是不是函數(shù)式接口:
- @FunctionalInterface
- 放在接口定義的上方:如果接口是函數(shù)式接口,編譯通過(guò),反之失敗。
注意:
- 我們自己定義函數(shù)式接口的時(shí)候,@FunctionalInterface是可選的,就算不寫,只要爆炸慢煮函數(shù)式接口定義的要求,照樣也是。但是建議加上注解。
1.2 函數(shù)式接口作為方法的參數(shù)
如果方法的參數(shù)是一個(gè)函數(shù)式接口,我們可以使用Lambda表達(dá)式作為參數(shù)傳遞
定義一個(gè)類(RunnableDemo).在類中提供倆個(gè)方法
一個(gè)方法是 startThread(Runnable r) 方法參數(shù)Runnable是一個(gè)函數(shù)式接口
一個(gè)方法是主方法.在主方法中凋用startThread方法
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
public class RunnableDemo { public static void main(String[] args) { //在主方法中凋用startThread方法 //匿名內(nèi)部類 startThread( new Runnable() { @Override public void run() { System.out.println(Thread.currentThread().getName() + " 線程啟動(dòng)" ); } }); //Lambda startThread(() -> System.out.println(Thread.currentThread().getName() + " 線程啟動(dòng)" )); } private static void startThread(Runnable r) { //方法參數(shù)Runnable是一個(gè)凾數(shù)式接口 // Thread t = new Thread(r); // t.start(); new Thread(r).start(); } } |
結(jié)果:
Thread-0 線程啟動(dòng)
Thread-1 線程啟動(dòng)
1.3 函數(shù)式接口作為方法的返回值
如果方法的返回值是一個(gè)函數(shù)式接口,我們可以使用Lambda表達(dá)式來(lái)作為結(jié)果返回
練習(xí):
定義一個(gè)類(ComparatorDemo).在類中提供倆個(gè)方法
一個(gè)方法是 Comparotorestring getComparator() 方法返回値Comparator是一個(gè)凾數(shù)式接口
一個(gè)方法是主方法.在主方法中調(diào)用getComparator方法
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
|
public class ComparatorDemo { public static void main(String[] args) { //構(gòu)造使用場(chǎng)景 根據(jù)字符串長(zhǎng)度 //定義集合,存儲(chǔ)字符串元素 ArrayList<String> array = new ArrayList<String>(); array.add( "Collin" ); array.add( "Leon" ); array.add( "Pi" ); array.add( "123" ); array.add( "878" ); System.out.println( "排序前: " + array); Collections.sort(array); System.out.println( "排序后: " + array); Collections.sort(array, getComparator()); System.out.println( "排序后: " + array); } private static Comparator<String> getComparator(){ //匿名內(nèi)部類的方式實(shí)現(xiàn) //根據(jù)長(zhǎng)度去排序 // Comparator<String > comp = new Comparator<String>() { // // @Override // public int compare(String o1, String o2) { // return o1.length() - o2.length(); // } // }; // return comp; //改進(jìn) // return new Comparator<String>() { // // @Override // public int compare(String o1, String o2) { // return o1.length() - o2.length(); // } // }; //改進(jìn)為L(zhǎng)ambda表達(dá)式 // return ((String s1, String s2) -> { // return s1.length()-s2.length(); // }); //優(yōu)化 return (s1,s2) -> s1.length()-s2.length(); } } |
結(jié)果:
排序前: [Collin, Leon, Pi, 123, 878]
排序后: [123, 878, Collin, Leon, Pi]
排序后: [Pi, 123, 878, Leon, Collin]
1.4常見(jiàn)的函數(shù)式接口
Java8在java.util.function包下預(yù)定義了大量的函數(shù)式接口
- 功能性接口 java.Util.Function<T,R>{public R apply(T t);} :接口一個(gè)參數(shù),返回一個(gè)參數(shù)
- 消費(fèi)性接口 Consumer{public void accept(T t);} :不需要返回
- 供給性接口 Supplier{public T get();}
- 斷言性接口 Predicate{public boolean test(T t);}: 判斷使用
1.5 Supplier接口
主要用來(lái)生產(chǎn)數(shù)據(jù)
Supplier< T >: 包含一個(gè)無(wú)參的方法
- T get():獲取結(jié)果
- 該方法不需要參數(shù),會(huì)按照某種實(shí)現(xiàn)邏輯(由Lambda表達(dá)式實(shí)現(xiàn))返回一個(gè)數(shù)據(jù)
- Supplier< T > 接口也被稱為生產(chǎn)型接口,如果我們制定了接口的泛型是什么類型,那么接口中的get方法就會(huì)生產(chǎn)什么類型的數(shù)據(jù)供我們使用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
public class SupplierDemo { public static void main(String[] args) { // String s = getstring(() -> { // return "你好Java"; // }); //優(yōu)化 String s = getstring(() -> "略略略" ); System.out.println(s); Integer i = getInteger(() -> 9920 ); System.out.println(i); } //定義一個(gè)方法,返回一個(gè)int數(shù)據(jù) private static Integer getInteger(Supplier<Integer> sup) { return sup.get(); } //定義一個(gè)方法,返回一個(gè)String數(shù)據(jù) private static String getstring(Supplier<String> sup) { return sup.get(); } } |
結(jié)果:
略略略
9920
1.6 Customer接口
Consumer< T > :包含倆個(gè)方法
- void accept( T t ):對(duì)指定的參數(shù)執(zhí)行此操作
- default Consumer< T >and Then( Consumer after ):返回一個(gè)組合的Consumer,依次執(zhí)行操作,然后執(zhí)行after操作
- 該接口也被稱為消費(fèi)型接口,它消費(fèi)的數(shù)據(jù)的類型由泛型指定
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
public static void main(String[] args) { operatorString( "Leo1" , (String s) ->{ System.out.println(s); }); //優(yōu)化 operatorString( "Leo2" , s -> System.out.println(s)); //方法引用 operatorString( "Leo3" , System.out::println); operatorString( "Leo4" , s -> System.out.println( new StringBuilder(s).reverse())); System.out.println( "--------------" ); operatorString( "Leon5" , s -> System.out.println(s),s -> System.out.println( new StringBuilder(s).reverse())); } //定一個(gè)方法,用不同的方式消費(fèi)同一個(gè)一個(gè)字符串?dāng)?shù)據(jù)倆次 private static void operatorString(String name, Consumer<String> con1,Consumer<String> con2) { // con1.accept(name); // con2.accept(name); con1.andThen(con2).accept(name); //先消費(fèi)name,再con2 } //定一個(gè)方法,消費(fèi)一個(gè)字符串?dāng)?shù)據(jù) private static void operatorString(String name, Consumer<String> con) { con.accept(name); } } |
結(jié)果:
Leo1
Leo2
Leo3
4oeL
--------------
Leon5
5noeL
練習(xí)
String[] strArray= {"林青霞30",”張曼玉,35", "王祖賢,33");
字符串?dāng)?shù)組中有多條信息, 請(qǐng)按照格式:“姓名: XX年齡: XX"的格式將信息打印出來(lái)
要求:
把打印姓名的動(dòng)作作為第一個(gè)Consumer接口的L ambda實(shí)例
把打印年齡的動(dòng)作作為第二個(gè)Consumer接[ ]的ambda實(shí)例
將兩個(gè)Consumer接口按照順序組合到一起使用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
public class ConsumerTest { public static void main(String[] args) { String[] strArray= { "林青霞,30" , "張曼玉,35" , "王祖賢,33" }; printInfo(strArray, (String str) ->{ String name = str.split( "," )[ 0 ]; System.out.print( "Name: " + name); },(String str) -> { int age =Integer.parseInt(str.split( "," )[ 1 ]); System.out.println( " ,Age: " + age); }); System.out.println( "-------------------" ); //改進(jìn) printInfo(strArray, str -> System.out.print( "Name: " + str.split( "," )[ 0 ]), str -> System.out.println( " ,Age: " + Integer.parseInt(str.split( "," )[ 1 ])) ); } private static void printInfo(String[] strArray, Consumer<String> con1, Consumer<String> con2) { for (String str : strArray) { con1.andThen(con2).accept(str); } } } |
結(jié)果:
Name: 林青霞 ,Age: 30
Name: 張曼玉 ,Age: 35
Name: 王祖賢 ,Age: 33
-------------------
Name: 林青霞 ,Age: 30
Name: 張曼玉 ,Age: 35
Name: 王祖賢 ,Age: 33
1.7Predicate接口
Predicate< T >:常用的四個(gè)方法
- boolean test(T t):對(duì)給定的參數(shù)進(jìn)行判斷(判斷邏輯由Lambda表達(dá)式實(shí)現(xiàn)),返回一個(gè)布爾值
- default Predicate negate():返回一個(gè)邏輯的否定,對(duì)應(yīng)邏輯非
- default Predicate and(Predicate other):返回一個(gè)組合判斷,對(duì)應(yīng)短路與
- default Predicate or(Predicate other):返回一個(gè)組合判斷,對(duì)應(yīng)短路或
- Predicate< T >接口通常用于判斷參數(shù)是否滿足指定的條件
練習(xí)
String[] strArray= {"林青霞,30","柳巖,34", "張曼玉,35","貂蟬,31","王祖賢,33"};
字符串?dāng)?shù)組中有多條信息,請(qǐng)通過(guò)Predicate接口的拼裝將符合要求的字符串篩選到集合ArrayList中,井遍歷ArrayList集合
同時(shí)滿足如下要求:
姓名長(zhǎng)度大于2;
年齡大于33
分析
有兩個(gè)判斷條件,所以需要使用兩個(gè)Predicate接[ ],對(duì)條件進(jìn)行判斷
必須同時(shí)滿足兩個(gè)條件,所以可以使用and方法連接兩個(gè)判斷條件
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
public class PredicateTest { public static void main(String[] args) { String[] strArray= { "林青霞,30" , "柳巖,34" , "張曼玉,35" , "貂蟬,31" , "王祖賢,33" }; ArrayList<String> array = myFilter(strArray, s -> s.split( "," )[ 0 ].length() > 2 , s -> Integer.parseInt(s.split( "," )[ 1 ]) > 33 ); for (String str : array) { System.out.println(str); } } //通過(guò)Predicate接口的拼裝將符合要求的字符串篩選到集合ArrayList中 private static ArrayList<String> myFilter(String[] strArray, Predicate<String> pre1, Predicate<String> pre2){ //定義一個(gè)集合 ArrayList<String> array = new ArrayList<String>(); //遍歷數(shù)組 for (String str : strArray) { if (pre1.and(pre2).test(str)) { array.add(str); } } return array; } } |
結(jié)果:
張曼玉,35
1.8 Function接口
Function<T,R>:常用的兩個(gè)方法
- R apply(T t): 將此函數(shù)應(yīng)用于給定的參數(shù)
- default < V > Function andThen (Function after):返回一個(gè)組合函數(shù),首先將該函數(shù)應(yīng)用輸入,然后將after函數(shù)作用于結(jié)果
- Function<T,R> 接口通常用于對(duì)參數(shù)進(jìn)行處理,轉(zhuǎn)換(處理邏輯由Lambda表達(dá)式實(shí)現(xiàn)),然后返回一個(gè)新的値
練習(xí)
String s =“林青霞30";
請(qǐng)按照我指定的要求進(jìn)行操作:
1:將字符串截取得到數(shù)字年齡部分
2:將上一步的年齡字符串轉(zhuǎn)換成為int類型的數(shù)據(jù)
3:將上-步的int數(shù)據(jù)加70, 得到一個(gè)int結(jié)果,在控制臺(tái)輸出
請(qǐng)通過(guò)Function接口來(lái)實(shí)現(xiàn)函數(shù)拼接
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
public class FunctionTest { public static void main(String[] args) { String s = "林青霞,30" ; convert(s, ss -> s.split( "," )[ 1 ], ss -> Integer.parseInt(ss), i -> i + 70 ); } private static void convert(String s, Function<String, String> fun1, Function<String, Integer> fun2, Function<Integer, Integer> fun3) { int i = fun1.andThen(fun2).andThen(fun3).apply(s); System.out.println(i); } } |
結(jié)果:
100
總結(jié)
到此這篇關(guān)于Java8函數(shù)式接口基礎(chǔ)學(xué)習(xí)教程的文章就介紹到這了,更多相關(guān)Java8函數(shù)式接口內(nèi)容請(qǐng)搜索服務(wù)器之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持服務(wù)器之家!
原文鏈接:https://juejin.cn/post/6949030236637888549