JDK 8 函數(shù)式接口 Supplier、Function、Consumer、Predicate
Posted on 2020-07-24 15:46 為自己代言 閱讀(1333) 評(píng)論(0) 編輯 收藏 所屬分類: java/J2EE函數(shù)式接口的特征
1、三種方法
- 唯一的抽象方法
- 使用default定義普通方法(默認(rèn)方法),通過對(duì)象調(diào)用。
- 使用static定義靜態(tài)方法,通過接口名調(diào)用。
2、一個(gè)新注解@FunctionInterface
- 注解@FunctionalInterface告訴編譯器這是一個(gè)函數(shù)式接口,明確這個(gè)函數(shù)中只有一個(gè)抽象方法,當(dāng)你嘗試在接口中編寫多個(gè)抽象方法的時(shí)候編譯器將不允許,但是可以有多個(gè)非抽象方法。
- 不過Object類的方法可以定義為抽象方法,因?yàn)榻涌诘膶?shí)現(xiàn)類一定是Object的子類
- 如果接口被標(biāo)注了@FunctionalInterface,這個(gè)類就必須符合函數(shù)式接口的規(guī)范
- 即使一個(gè)接口沒有標(biāo)注@FunctionalInterface,如果這個(gè)接口滿足函數(shù)式接口規(guī)則,依舊被當(dāng)作函數(shù)式接口。
3、JDK 1.8 新增加的函數(shù)接口包:
java.util.function.*
java.util.function 它包含了很多接口,用來支持 Java的 函數(shù)式編程,它們大致分為五類:
/**
*JDK 8 函數(shù)式接口 Supplier、Function、Consumer、Predicate
*JDK 8 函數(shù)式接口 Supplier、Function、Consumer、Predicate
* @param args
* @throws Exception
*/
public static void main(String[] args) throws Exception {
ThreadPoolExecutor executor = (ThreadPoolExecutor)newFixedThreadPool(10);
//1:JDK8以前,通過匿名內(nèi)部類實(shí)現(xiàn)函數(shù)式接口
executor.submit(new Runnable() {
@Override
public void run() {
System.out.println("JDK8以前,通過匿名內(nèi)部類實(shí)現(xiàn)函數(shù)式接口");
}
});
//2:JDK8以后可以使用lambda 表達(dá)式來實(shí)現(xiàn),lambda表達(dá)式就是為了優(yōu)化匿名內(nèi)部類而生(分開寫效果)
Thread thread = new Thread(() -> System.out.println("task running !"));
Runnable r = () -> System.out.println("JDK8以后可以使用lambda 表達(dá)式來實(shí)現(xiàn),lambda表達(dá)式就是為了優(yōu)化匿名內(nèi)部類而生(分開寫效果)!");
executor.submit(r);
//3:合并起來效果
executor.submit(() -> {
System.out.println("JDK8以后可以使用lambda 表達(dá)式來實(shí)現(xiàn),lambda表達(dá)式就是為了優(yōu)化匿名內(nèi)部類而生!");
});
//4:其它 Supplier、Function、Consumer、Predicate 都可以用lambda 表達(dá)式來實(shí)現(xiàn)
Supplier<String> supplier = () -> "我是SuSupplier";
Supplier<Integer> supplier2 = () -> new Integer(1);
System.out.println("supplier=" + supplier.get() + ";supplier2=" + supplier2.get());
//5: Function功能型函數(shù)式接口 Function<T, R> 接受一個(gè)輸入?yún)?shù)T,返回一個(gè)結(jié)果R
Function<String,Integer> function=str -> Integer.parseInt(str);
Function<Integer,String> fun2 = item -> item+"";
System.out.println("輸入字符型 1 返回int型結(jié)果:"+function.apply("1"));
System.out.println("輸入整型 1 返回字符型結(jié)果:"+fun2.apply(2));
//6: Consumer 一個(gè)接受單個(gè)輸入?yún)?shù)并且不返回結(jié)果的操作。 與大多數(shù)其他函數(shù)接口不同, Consumer接口期望通過接受參數(shù),改普通對(duì)象引用值(說明白點(diǎn)就是對(duì)原來的值進(jìn)行加工,注意返回值 void)
Consumer<StringBuffer> consumer= sb->sb.append("-yyy");
StringBuffer sb1=new StringBuffer().append("111");
consumer.accept(sb1);
//改變sb的內(nèi)部引用值
System.out.println("=========s="+sb1.toString());
//7: Predicate<T> 斷言型接口常用于集合的過濾,得到一個(gè)新的集合 Stream filter(Predicate<? super T> predicate);
Predicate<Integer> predicate = age -> age > 18;
Predicate<String> predicate2 = str -> str != null;
System.out.println(predicate.test(19));
System.out.println(predicate2.test(null));
//我們常用集合過濾類就是對(duì)這個(gè)接口實(shí)現(xiàn)類 其中 filter(Predicate<? super T> predicate) 用的就是這個(gè)接口
List<String> list= Lists.newArrayList("1","2","2","3","4","4","8");
list.stream().map(s -> Long.parseLong(s)).distinct().filter(s -> s < 10).collect(Collectors.toList()).forEach(u -> System.out.println(u));
//總結(jié),以上的例子其實(shí)都是JDK8 lambda 表達(dá)式簡潔的寫法,而且全是合并寫的,并沒有分開步驟寫(所有函數(shù)性接口,都可以用lambda 表達(dá)式簡潔寫法)
//關(guān)閉線程池
executor.shutdownNow();
}