函数式接口:有且仅有一个抽象方法的接口 Java中的函数式编程体现就是Lambda表达式,所以函数式接口就是可以使用与Lambda使用的接口 只有确保接口只能够有且只有一个抽象方法,Lambda才能顺利的进行推导
检测接口是不是函数式接口:
@FunctionalInterface放在接口定义的上方:如果接口是函数式接口,编译通过,反之失败。注意:
我们自己定义函数式接口的时候,@FunctionalInterface是可选的,就算不写,只要爆炸慢煮函数式接口定义的要求,照样也是。但是建议加上注解。如果方法的参数是一个函数式接口,我们可以使用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 线程启动如果方法的返回值是一个函数式接口,我们可以使用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]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);}: 判断使用主要用来生产数据 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(); } } 结果: 略略略 9920Consumer< 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 5noeLPredicate< 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; } } 结果: 张曼玉,35Function<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