App下載

Java基礎(chǔ)學(xué)習(xí) 關(guān)于函數(shù)式接口的學(xué)習(xí)教程

猿友 2021-07-21 11:31:58 瀏覽數(shù) (1967)
反饋

在 Java 中,函數(shù)式接口有且僅有一個(gè)抽象方法。我們熟知的 lambda 表達(dá)式其實(shí)就是函數(shù)式接口的一種簡(jiǎn)單實(shí)現(xiàn)方法。下面我將和大家一起學(xué)習(xí) Java 中的函數(shù)式接口的詳細(xì)內(nèi)容。

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是可選的,就算不寫(xiě),只要爆炸慢煮函數(shù)式接口定義的要求,照樣也是。但是建議加上注解。

1.2 函數(shù)式接口作為方法的參數(shù)

如果方法的參數(shù)是一個(gè)函數(shù)式接口,我們可以使用Lambda表達(dá)式作為參數(shù)傳遞

定義一個(gè)類(lèi)(RunnableDemo).在類(lèi)中提供倆個(gè)方法

 一個(gè)方法是 startThread(Runnable r) 方法參數(shù)Runnable是一個(gè)函數(shù)式接口

 一個(gè)方法是主方法.在主方法中凋用startThread方法

public class RunnableDemo {
	public static void main(String[] args) {
		//在主方法中凋用startThread方法
		
		//匿名內(nèi)部類(lèi)
		startThread(new Runnable() {
			
			@Override
			public void run() {
				System.out.println(Thread.currentThread().getName() + " 線(xiàn)程啟動(dòng)");
			}
		});
		
		//Lambda
		startThread(() -> System.out.println(Thread.currentThread().getName() + " 線(xiàn)程啟動(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 線(xiàn)程啟動(dòng)
Thread-1 線(xiàn)程啟動(dòng)

1.3 函數(shù)式接口作為方法的返回值

如果方法的返回值是一個(gè)函數(shù)式接口,我們可以使用Lambda表達(dá)式來(lái)作為結(jié)果返回

練習(xí):
定義一個(gè)類(lèi)(ComparatorDemo).在類(lèi)中提供倆個(gè)方法
 一個(gè)方法是 Comparotorestring getComparator() 方法返回値Comparator是一個(gè)凾數(shù)式接口
 一個(gè)方法是主方法.在主方法中調(diào)用getComparator方法

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)部類(lè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ēng)為生產(chǎn)型接口,如果我們制定了接口的泛型是什么類(lèi)型,那么接口中的get方法就會(huì)生產(chǎn)什么類(lèi)型的數(shù)據(jù)供我們使用
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操作
  • 該接口也被稱(chēng)為消費(fèi)型接口,它消費(fèi)的數(shù)據(jù)的類(lèi)型由泛型指定
	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接口按照順序組合到一起使用

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ù)是否滿(mǎn)足指定的條件

  練習(xí)

  String[] strArray= {"林青霞,30","柳巖,34", "張曼玉,35","貂蟬,31","王祖賢,33"};

  字符串?dāng)?shù)組中有多條信息,請(qǐng)通過(guò)Predicate接口的拼裝將符合要求的字符串篩選到集合ArrayList中,井遍歷ArrayList集合

  同時(shí)滿(mǎn)足如下要求:

   姓名長(zhǎng)度大于2;

   年齡大于33

 分析

  有兩個(gè)判斷條件,所以需要使用兩個(gè)Predicate接[ ],對(duì)條件進(jìn)行判斷

  必須同時(shí)滿(mǎn)足兩個(gè)條件,所以可以使用and方法連接兩個(gè)判斷條件

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類(lèi)型的數(shù)據(jù)

 3:將上-步的int數(shù)據(jù)加70, 得到一個(gè)int結(jié)果,在控制臺(tái)輸出

 請(qǐng)通過(guò)Function接口來(lái)實(shí)現(xiàn)函數(shù)拼接

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)于 Java 基礎(chǔ),函數(shù)式接口的基礎(chǔ)學(xué)習(xí)教程的文章就介紹到這了,想要了解更多相關(guān) Java 函數(shù)式接口的其他內(nèi)容請(qǐng)搜索W3Cschool以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持!


0 人點(diǎn)贊