lambda表达式,看完了你就是大佬了!

前面一篇文章介绍了两个经常使用函数式接口SupplierConsumer接口,今天接着讲另外另个重要函数式接口PredicateFunction接口。这几个接口在Java中应用普遍,比方Stream流式API中就常常用到。java

2. Predicate接口

当咱们须要对某种数据类型进行判断,获得一个boolean值结果时候,能够采用java.util.function.Predicate<T>接口。比方说对传进来的List判断是否包含某个特定字符PHP。app

2.1 条件判断

咱们有时候须要进行业务判断,假设我要实现一个功能入参为List,若是List参数知足某个条件则输出对应的结果。比方说下面的代码checkchar()若是入参list1知足某个条件那么输出System.out.println("知足对应条件"),具体条件是什么还不肯定,多是判断List中是否包含某个元素,大小是否符合要求。具体由调用方来实现。ide

private static boolean checkChar(List list1, Predicate<List> predicateList) {
        boolean rs = predicateList.test(list1);
        if (rs) {
            System.out.println("知足对应的条件");
        }
        return rs;
    }

下面咱们实现main方法调用上面checkchar()方法,而且经过lambda表达式来实现里面具体判断条件。函数

public static void main(String[] args) {
        List list1 = new ArrayList();
        list1.add("PHP");
        list1.add("JAVA");
        list1.add("C++");
        boolean rs = checkChar(list1, s -> {
            return s.contains("PHP");
        });

        if (rs) {
            System.out.println("list1中包含PHP 字符");
        } else {
            System.out.println("list1中不包含PHP 字符");
        }
    }

main方法中,调用checkChar()方法的时候,传入lambda表达式s-&gt;{return s.contains("PHP")}来进行判断。这意思就是说在调用checkChar方法的时候若是list1列表中包含PHP这个元素执行System.out.println("知足对应的条件")语句。ui

上述代码执行结果为:this

知足对应的条件
list1中包含PHP 字符

2.2 与判断

上面咱们说了单个条件判断,若是想多个条件判断怎么办?可使用Predicate的and()方法。code

咱们先定义一个函数,这个函数里会进行两个判断,两个判断具体的逻辑由外部传过来。对象

private static boolean checkChar2(List list1, Predicate<List> pre, Predicate<List> other) {
        return pre.and(other).test(list1);
}

main方法中,传入两个判断逻辑s -&gt; { return s.contains("PHP");}s -&gt; { return s.contains("C++");},而后经过Predicate.and()方法执行与操做。接口

public static void main(String[] args) {
        List list1 = new ArrayList();
        list1.add("PHP");
        list1.add("JAVA");
        list1.add("C++");
        boolean rs = checkChar2(list1, s -> {
            return s.contains("PHP");
        }, s -> {
            return s.contains("C++");
        });

        if (rs) {
            System.out.println("list1同时有PHP和C++字符");
        } else {
            System.out.println("list1不知足同时有PHP和C++字符");
        }
    }

上述代码的执行结果为:three

list1同时有PHP和C++字符

2.3 非判断

"与"、"或"咱们已经了解了,剩下还有一个"非"(取反)操做,咱们看一下取反操做negate如何使用。

Predicatenegate()方法以下:

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

从源码咱们能够看到negate方法是在执行了test()方法以后,对结果boolean值进行取反而已。请注意,必定要先调用negate方法而后调用test方法 ,这个跟andor方法同样:

private static boolean checkNotChar(List list1, Predicate<List> pre1) {
        return pre1.negate().test(list1);
}

public static void main(String[] args) {
        List list1 = new ArrayList();
        list1.add("PHP");
        list1.add("JAVA");
        list1.add("C++");
        boolean rs = checkNotChar(list1, s -> {
            return s.contains("C#");
        });

        if (rs) {
            System.out.println("list1没有C#");
        } else {
            System.out.println("list1有C#");
        }
}

上述代码执行结果为:

list1没有C#

3. Function接口

java.util.function.Function&lt;T,R&gt;至关于数据中的函数,一个类型的数据做为输入获得另外一个类型数据的输出。

3.1 转换方法:apply

Function中执行转换的方法为抽象方法R apply(T t),该方法根据参数T类型数据获取类型为R的结果。好比下面我要将String类型转化为Integer类型。具体如何转换由调用方经过lambda来决定。

咱们先定义转换的方法:

private static Integer transfer(String param, Function<String, Integer> function) {
        int num = function.apply(param);
        return num;
}

具体转换逻辑在main方法中定义:

public static void main(String[] args) {
        String str = "99";
        int num = transfer(str, s -> (Integer.parseInt(s) + 1));//这里定义具体的转换逻辑
        System.out.println("DemoFunc执行结果为" + num);
}

3.2 级联转换:andThen

若是咱们想作多步转换那么就须要用到andThen()方法了,这些具体转换的实如今调用放经过lambda来实现。

我先定义一个级联转换的方法chainTransfer,该方法中传入三个Function依次通过one,two,three进行转换,而后返回结果。

private static Integer chainTransfer(String str, Function<String, Integer> one, Function<Integer, Integer> two,
        Function<Integer, Integer> three) {
        int num = one.andThen(two).andThen(three).apply(str);
        return num;
    }

下面在main方法中实现3个Function而后调用转换函数chainTransfer

public static void main(String[] args) {
        int num = chainTransfer("9", str -> Integer.parseInt(str) + 10, 
                                i -> i *= 10, i -> i + 5);
        System.out.println("转换后的结果为:" + num);
    }

咱们执行上面main方法结果为:

转换后的结果为:195

具体转换步骤为:

1. 先将9转换成Integer类型而后+10,对应Integer.parseInt(str) + 10,获得19;
2. 而后乘以10,获得19*10 = 190;
3. 第三部加5,获得190+5 = 195。

固然定义这个方法的时候有一个要注意的,依赖被调用的Function参数类型要是适配,比方说Function&lt;String,Integer&gt; one入参为String返回值为Integer,那么后面Function two的入参必须是Integer。即前一个Function的返回值跟后一个Function入参类型应该兼容。

3.2 讨论级联转换中的类型兼容问题

前面咱们说了,即前一个Function的返回值跟后一个Function入参类型应该兼容。由于前一个Function是后一个Function的入参,在函数调用中当实际入参是形参的子类那么是兼容的。

即以下的转换函数式兼容的:

private static User objChainTransfer(String str, Function<String, VipUser> one, Function<User, User> two) {
        User u = one.andThen(two).apply(str);
        return u;
    }
}

前面一个Function返回的是VipUser,其为第二个Function的入参User的子类。咱们在main函数中调用上面的函数,其代码以下:

public static void main(String[] args) {
        User u = objChainTransfer("19", str -> {
            return new VipUser("name" + str, str);
        }, vipUser -> {
            vipUser.setName(vipUser.getName() + "_new");
            return vipUser;
        });

        System.out.println("转换后的结果为:" + u);
}

运行后其结果为:

转换后的结果为:VipUser{name='name19_new', id='19'}

这段代码作了以下事情:

1. 咱们先建立一个`VipUser`对象
2. 修改name属性,
3. 返回User对象。

进一步咱们能够分析源码:

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

源码中,andThen方法中Function&lt;? super R, ? extends V&gt; after,表示after的这个Function的入参必须是R的超类,其中R是第一个Function的返回值。

具体能够查看Function.java的源码:

public interface Function<T, R> {
    /**
     * Applies this function to the given argument.
     *
     * @param t the function argument
     * @return the function result
     */
    R apply(T t);
    //...
}
相关文章
相关标签/搜索