javaweb-函数式接口-018

第一章 函数式接口

函数式接口的概念&函数式接口的定义

1.1 概念
函数式接口在Java中是指:有且仅有一个抽象方法的接口。
函数式接口,即适用于函数式编程场景的接口。而Java中的函数式编程体现就是Lambda,所以函数式接口就是可
以适用于Lambda使用的接口。只有确保接口中有且仅有一个抽象方法,Java中的Lambda才能顺利地进行推导。
备注:“语法糖”是指使用更加方便,但是原理不变的代码语法。例如在遍历集合时使用的for-each语法,其实
底层的实现原理仍然是迭代器,这便是“语法糖”。从应用层面来讲,Java中的Lambda可以被当做是匿名内部
类的“语法糖”,但是二者在原理上是不同的。
1.2 格式
只要确保接口中有且仅有一个抽象方法即可:
修饰符 interface 接口名称 {
public abstract 返回值类型 方法名称(可选参数信息);
// 其他非抽象方法内容
}

函数式接口:有且只有一个抽象方法的接口,称之为函数式接口
当然接口中可以包含其他的方法(默认,静态,私有)
@FunctionalInterface注解
作用:可以检测接口是否是一个函数式接口,是一个检测者
是:编译成功,注解不报错
否:编译失败(接口中没有抽象方法 抽象方法的个数多余1个),注解报错

@FunctionalInterface
public interface MyFunctionalInterface {
    //定义一个抽象方法
    public abstract void method();
}

如果这个接口不是函数式接口,注解报错

函数式接口的使用

可以使用匿名内部类,lambda

第二章 函数式编程

性能浪费的日志案例

lamda不创建类class文件,少了调取文件的步骤,比匿名内部类效率高一些
假设存在三个字符串,要求如果满足条件则拼接,
那么就可以使用lambda实现延迟加载

使用Lambda优化日志案例

实际上使用内部类也可以达到同样的效果,只是将代码操作延迟到了另外一个对象当中通过调用方法
来完成。而是否调用其所在方法是在条件判断之后才执行的。
如果条件不满足,字符串就不用链接了,不用先拼接字符串再判断显示条件

public class MyMain {
    public static void main(String[] args) {
        //延迟执行
        int a = 3;
        String b = "123";
        String c = "456";
        hello(a ,new MyInter() {
            @Override
            public void run() {
                System.out.println("执行");
                System.out.println(b+c);
            }
        });
    }

    public static void hello(int i,MyInter myInter) {
        if (i>4) {
            myInter.run();
        }
    }
}

函数式接口作为方法的参数案例

这里的lambda作为执行体存在,事实上因为函数式接口可以当参数和返回值,那么lambda同样可以兼职
如果方法的参数是一个函数式接口类型,那么就可以使用Lambda表达式进行替代。使用Lambda表达式作为方法参数,其实就是使用函数式接口作为方法参数。

 public static void hello(int i,MyInter myInter) {
        if (i>4) {
            myInter.run();
        }
    }
  hello(3,()-> System.out.println("你好"));

函数式接口作为方法的返回值类型案例

类似地,如果一个方法的返回值类型是一个函数式接口,那么就可以直接返回一个Lambda表达式。

接口作为返回值,实际得到的是一个对象

public static MyInter world() {
        return () -> System.out.println("你好,改变!");
    }
`world().run();``

第三章 常用函数式接口

JDK提供了大量常用的函数式接口以丰富Lambda的典型使用场景,它们主要在 java.util.function 包中被提供。
下面是最简单的几个接口及使用示例。

常用的函数式接口_Supplier接口

java.util.function.Supplier 接口仅包含一个无参的方法: T get() 。用来获取一个泛型参数指定类型的对
象数据。由于这是一个函数式接口,这也就意味着对应的Lambda表达式需要“对外提供”一个符合泛型类型的对象
数据。

常用的函数式接口_Consumer接口

java.util.function.Consumer 接口则正好与Supplier接口相反,它不是生产一个数据,而是消费一个数据,
其数据类型由泛型决定。
Consumer 接口中包含抽象方法 void accept(T t) ,意为消费一个指定泛型的数据。

常用的函数式接口_Consumer接口的默认方法andThen`

默认方法:andThen
如果一个方法的参数和返回值全都是 Consumer 类型,那么就可以实现效果:消费数据的时候,首先做一个操作,
然后再做一个操作,实现组合。而这个方法就是 Consumer 接口中的default方法 andThen

default Consumer<T> andThen(Consumer<? super T> after) { 
		Objects.requireNonNull(after);
		 return (T t)> { accept(t); after.accept(t); }; 
 }

即是a调用accept方法后,交给b再次处理

java.util.Objects 的 requireNonNull 静态方法将会在参数为null时主动抛出 NullPointerException 异常。这省去了重复编写if语句和抛出空指针异常的麻烦。
要想实现组合,需要两个或多个Lambda表达式即可,而 andThen 的语义正是“一步接一步”操作。

常用的函数式接口_Predicate接口

有时候我们需要对某种类型的数据进行判断,从而得到一个boolean值结果。这时可以使用 java.util.function.Predicate 接口。
java.util.function.Predicate接口
作用:对某种数据类型的数据进行判断,结果返回一个boolean值
Predicate接口中包含一个抽象方法:
boolean test(T t):用来对指定数据类型数据进行判断的方法
结果:
符合条件,返回true
不符合条件,返回false

常用的函数式接口_Predicate接口_默认方法and

既然是条件判断,就会存在与、或、非三种常见的逻辑关系。其中将两个 Predicate 条件使用“与”逻辑连接起来实
现“并且”的效果时,可以使用default方法 and 。

default Predicate<T> and(Predicate<? super T> other) {
Objects.requireNonNull(other);
return (t)> test(t) && other.test(t);
}

逻辑表达式:可以连接多个判断的条件
&&:与运算符,有false则false
||:或运算符,有true则true
!:非(取反)运算符,非真则假,非假则真

常用的函数式接口_Predicate接口_默认方法or&negate

与 and 的“与”类似,默认方法 or 实现逻辑关系中的“或”。JDK源码为:

default Predicate<T> or(Predicate<? super T> other) { 
		Objects.requireNonNull(other); 
		return (t)> test(t) || other.test(t); 
}

取反
默认方法:negate

default Predicate<T> negate() {
return (t)> !test(t);
}

17_常用的函数式接口_Function接口


/*
    java.util.function.Function<T,R>接口用来根据一个类型的数据得到另一个类型的数据,
        前者称为前置条件,后者称为后置条件。
    Function接口中最主要的抽象方法为:R apply(T t),根据类型T的参数获取类型R的结果。
        使用的场景例如:将String类型转换为Integer类型。
 */
public class Demo01Function {
    /*
        定义一个方法
        方法的参数传递一个字符串类型的整数
        方法的参数传递一个Function接口,泛型使用<String,Integer>
        使用Function接口中的方法apply,把字符串类型的整数,转换为Integer类型的整数
     */
    public static void change(String s, Function<String,Integer> fun){
        //Integer in = fun.apply(s);
        int in = fun.apply(s);//自动拆箱 Integer->int
        System.out.println(in);
    }

    public static void main(String[] args) {
        //定义一个字符串类型的整数
        String s = "1234";
        //调用change方法,传递字符串类型的整数,和Lambda表达式
        change(s,(String str)->{
            //把字符串类型的整数,转换为Integer类型的整数返回
            return Integer.parseInt(str);
        });
        //优化Lambda
        change(s,str->Integer.parseInt(str));
    }
}

常用的函数式接口_Function接口_默认方法andThen

java.util.function.Function<T,R> 接口用来根据一个类型的数据得到另一个类型的数据,前者称为前置条件,
后者称为后置条件。
抽象方法:apply
Function 接口中最主要的抽象方法为: R apply(T t) ,根据类型T的参数获取类型R的结果。
Function 接口中有一个默认的 andThen 方法,用来进行组合操作。JDK源代码如:

default <V> Function<T, V> andThen(Function<? super R, ? extends V> after) {
Objects.requireNonNull(after);
return (T t)> after.apply(apply(t));
}

请注意,Function的前置条件泛型和后置条件泛型可以相同。

public class OffenFunctionInter {
    static String getString(Supplier<String> supplier) {
        return supplier.get();
    }

    static void useConsumer(Consumer<String> consumer) {
        consumer.accept("我是李玉成");
    }

    static void userAndThen(Consumer<String> consumer1, Consumer<String> consumer2) {
        consumer1.andThen(consumer2).accept("你好吗?");
    }

    static boolean usePredicate(Predicate<String> predicate) {
        return predicate.test("good good study,day day up");
    }

    static boolean useAnd(Predicate<String> predicate1, Predicate<String> predicate2) {
        return predicate1.and(predicate2).test("你好");
    }
    static boolean useOr(Predicate<String> predicate1, Predicate<String> predicate2) {
        return predicate1.or(predicate2).test("你好");
    }

    static boolean useNegate(Predicate<String> predicate) {
        return predicate.negate().test("你好");
    }

    static void useFunction(Function<Integer,String> function) {
        String apply = function.apply(100);
        System.out.println(apply.getClass());
    }

    static void useAndFunction(Function<String, Integer> function1, Function<Integer, Integer> function2) {
        Integer apply = function1.andThen(function2).apply("40");
        System.out.println(apply);
    }
    public static void main(String[] args) {
        System.out.println(getString(() -> "你好!" + "很高兴见到你"));
        useConsumer((string)-> System.out.println(string+"-》你呢?"));
        userAndThen((s) -> System.out.println(s + "我是李雷,我很好"), (s) -> System.out.println(s+"我是韩梅梅,我很好"));
        System.out.println(usePredicate((s) -> s.length() > 3));
        System.out.println(useAnd((s) -> s.length() > 1, (s) -> s.contains("你")));
        System.out.println(useOr((s) -> s.length() > 8, (s) -> s.contains("你")));
        System.out.println(useNegate((s) -> s.length() < 3));
        useFunction((i)->i+"");
        useAndFunction(((s) -> Integer.parseInt(s)),(i)->i+20);
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值