在 Java 中,函数式接口有且仅有一个抽象方法。我们熟知的 lambda 表达式其实就是函数式接口的一种简单实现方法。下面我将和大家一起学习 Java 中的函数式接口的详细内容。
1.1 函数式接口概述
函数式接口:有且仅有一个抽象方法的接口
Java中的函数式编程体现就是Lambda表达式,所以函数式接口就是可以使用与Lambda使用的接口
只有确保接口只能够有且只有一个抽象方法,Lambda才能顺利的进行推导
检测接口是不是函数式接口:
- @FunctionalInterface
- 放在接口定义的上方:如果接口是函数式接口,编译通过,反之失败。
注意:
- 我们自己定义函数式接口的时候,@FunctionalInterface是可选的,就算不写,只要爆炸慢煮函数式接口定义的要求,照样也是。但是建议加上注解。
1.2 函数式接口作为方法的参数
如果方法的参数是一个函数式接口,我们可以使用Lambda表达式作为参数传递
定义一个类(RunnableDemo).在类中提供俩个方法
一个方法是 startThread(Runnable r) 方法参数Runnable是一个函数式接口
一个方法是主方法.在主方法中凋用startThread方法
public class RunnableDemo {
public static void main(String[] args) {
//在主方法中凋用startThread方法
//匿名内部类
startThread(new Runnable() {
@Override
public void run() {
System.out.println(Thread.currentThread().getName() + " 线程启动");
}
});
//Lambda
startThread(() -> System.out.println(Thread.currentThread().getName() + " 线程启动"));
}
private static void startThread(Runnable r) {//方法参数Runnable是一个凾数式接口
// Thread t = new Thread(r);
// t.start();
new Thread(r).start();
}
}
结果:
Thread-0 线程启动
Thread-1 线程启动
1.3 函数式接口作为方法的返回值
如果方法的返回值是一个函数式接口,我们可以使用Lambda表达式来作为结果返回
练习:
定义一个类(ComparatorDemo).在类中提供俩个方法
一个方法是 Comparotorestring getComparator() 方法返回値Comparator是一个凾数式接口
一个方法是主方法.在主方法中调用getComparator方法
public class ComparatorDemo {
public static void main(String[] args) {
//构造使用场景 根据字符串长度
//定义集合,存储字符串元素
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(){
//匿名内部类的方式实现
//根据长度去排序
// Comparator<String > comp = new Comparator<String>() {
//
// @Override
// public int compare(String o1, String o2) {
// return o1.length() - o2.length();
// }
// };
// return comp;
//改进
// return new Comparator<String>() {
//
// @Override
// public int compare(String o1, String o2) {
// return o1.length() - o2.length();
// }
// };
//改进为Lambda表达式
// return ((String s1, String s2) -> {
// return s1.length()-s2.length();
// });
//优化
return (s1,s2) -> s1.length()-s2.length();
}
}
结果:
排序前: [Collin, Leon, Pi, 123, 878]
排序后: [123, 878, Collin, Leon, Pi]
排序后: [Pi, 123, 878, Leon, Collin]
1.4常见的函数式接口
Java8在java.util.function包下预定义了大量的函数式接口
- 功能性接口 java.Util.Function<T,R>{public R apply(T t);} :接口一个参数,返回一个参数
- 消费性接口 Consumer{public void accept(T t);} :不需要返回
- 供给性接口 Supplier{public T get();}
- 断言性接口 Predicate{public boolean test(T t);}: 判断使用
1.5 Supplier接口
主要用来生产数据
Supplier< T >: 包含一个无参的方法
- T get():获取结果
- 该方法不需要参数,会按照某种实现逻辑(由Lambda表达式实现)返回一个数据
- Supplier< T > 接口也被称为生产型接口,如果我们制定了接口的泛型是什么类型,那么接口中的get方法就会生产什么类型的数据供我们使用
public class SupplierDemo {
public static void main(String[] args) {
// String s = getstring(() -> {
// return "你好Java";
// });
//优化
String s = getstring(() -> "略略略");
System.out.println(s);
Integer i = getInteger(() -> 9920);
System.out.println(i);
}
//定义一个方法,返回一个int数据
private static Integer getInteger(Supplier<Integer> sup) {
return sup.get();
}
//定义一个方法,返回一个String数据
private static String getstring(Supplier<String> sup) {
return sup.get();
}
}
结果:
略略略
9920
1.6 Customer接口
Consumer< T > :包含俩个方法
- void accept( T t ):对指定的参数执行此操作
- default Consumer< T >and Then( Consumer after ):返回一个组合的Consumer,依次执行操作,然后执行after操作
- 该接口也被称为消费型接口,它消费的数据的类型由泛型指定
public static void main(String[] args) {
operatorString("Leo1", (String s) ->{
System.out.println(s);
});
//优化
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()));
}
//定一个方法,用不同的方式消费同一个一个字符串数据俩次
private static void operatorString(String name, Consumer<String> con1,Consumer<String> con2) {
// con1.accept(name);
// con2.accept(name);
con1.andThen(con2).accept(name);
//先消费name,再con2
}
//定一个方法,消费一个字符串数据
private static void operatorString(String name, Consumer<String> con) {
con.accept(name);
}
}
结果:
Leo1
Leo2
Leo3
4oeL
--------------
Leon5
5noeL
练习
String[] strArray= {"林青霞30",”张曼玉,35", "王祖贤,33");
字符串数组中有多条信息, 请按照格式:“姓名: XX年龄: XX"的格式将信息打印出来
要求:
把打印姓名的动作作为第一个Consumer接口的L ambda实例
把打印年龄的动作作为第二个Consumer接[ ]的ambda实例
将两个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("-------------------");
//改进
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);
}
}
}
结果:
Name: 林青霞 ,Age: 30
Name: 张曼玉 ,Age: 35
Name: 王祖贤 ,Age: 33
-------------------
Name: 林青霞 ,Age: 30
Name: 张曼玉 ,Age: 35
Name: 王祖贤 ,Age: 33
1.7Predicate接口
Predicate< T >:常用的四个方法
- boolean test(T t):对给定的参数进行判断(判断逻辑由Lambda表达式实现),返回一个布尔值
- default Predicate negate():返回一个逻辑的否定,对应逻辑非
- default Predicate and(Predicate other):返回一个组合判断,对应短路与
- default Predicate or(Predicate other):返回一个组合判断,对应短路或
- Predicate< T >接口通常用于判断参数是否满足指定的条件
练习
String[] strArray= {"林青霞,30","柳岩,34", "张曼玉,35","貂蝉,31","王祖贤,33"};
字符串数组中有多条信息,请通过Predicate接口的拼装将符合要求的字符串筛选到集合ArrayList中,井遍历ArrayList集合
同时满足如下要求:
姓名长度大于2;
年龄大于33
分析
有两个判断条件,所以需要使用两个Predicate接[ ],对条件进行判断
必须同时满足两个条件,所以可以使用and方法连接两个判断条件
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);
}
}
//通过Predicate接口的拼装将符合要求的字符串筛选到集合ArrayList中
private static ArrayList<String> myFilter(String[] strArray, Predicate<String> pre1, Predicate<String> pre2){
//定义一个集合
ArrayList<String> array = new ArrayList<String>();
//遍历数组
for (String str : strArray) {
if(pre1.and(pre2).test(str)) {
array.add(str);
}
}
return array;
}
}
结果:
张曼玉,35
1.8 Function接口
Function<T,R>:常用的兩个方法
- R apply(T t): 将此函数应用于给定的参数
- default < V > Function andThen (Function after):返回一个组合函数,首先将该函数应用输入,然后将after函数作用于結果
- Function<T,R> 接口通常用于对参数进行处理,转换(处理逻辑由Lambda表达式实现),然后返回一个新的値
练习
String s =“林青霞30";
请按照我指定的要求进行操作:
1:将字符串截取得到数字年龄部分
2:将上一步的年龄字符串转换成为int类型的数据
3:将上-步的int数据加70, 得到一个int结果,在控制台输出
请通过Function接口来实现函数拼接
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);
}
}
结果:
100
总结
到此这篇关于 Java 基础,函数式接口的基础学习教程的文章就介绍到这了,想要了解更多相关 Java 函数式接口的其他内容请搜索W3Cschool以前的文章或继续浏览下面的相关文章,希望大家以后多多支持!