lambda 表达式 (4)(Shawn),开发 android
上述代码执行结果为:
满足对应的条件 list1 中包含 PHP 字符
2.2 与判断
上面我们说了单个条件判断,如果想多个条件判断怎么办?可以使用Predicate的and()
方法。
我们先定义一个函数,这个函数里会进行两个判断,两个判断具体的逻辑由外部传过来。
private static boolean checkChar2(List list1, Predicate<List> pre, Predicate<List> other) {return pre.and(other).test(list1);}
main
方法中,传入两个判断逻辑 s -> { return s.contains("PHP");}
和 s -> { 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++字符");}}
上述代码的执行结果为:
list1 同时有 PHP 和 C++字符
2.3 非判断
"与"、"或"我们已经了解了,剩下还有一个"非"(取反)操作,我们看一下取反操作negate
如何使用。
Predicate
的negate()
方法如下:
default Predicate<T> negate() {return (t) -> !test(t);}
从源码我们可以看到negate
方法是在执行了test()
方法之后,对结果boolean
值进行取反而已。请注意,一定要先调用negate
方法然后调用test
方法 ,这个跟and
和or
方法一样:
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<T,R>
相当于数据中的函数,一个类型的数据作为输入得到另一个类型数据的输出。
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
具体转换步骤为:
先将 9 转换成 Integer 类型然后+10,对应 Integer.parseInt(str) + 10,得到 19;
然后乘以 10,得到 19*10 = 190;
第三部加 5,得到 190+5 = 195。
当然定义这个方法的时候有一个要注意的,依赖被调用的Function
参数类型要是适配,比方说Function<String,Integer> one
入参为String
返回值为Integer
,那么后面Function two
的入参必须是Integer
。即前一个Function
的返回值跟后一个Function
入参类型应该兼容。
3.2 讨论级联转换中的类型兼容问题
前面我们说了,即前一个Function
的返回值跟后一个Function
入参类型应该兼容。因为前一个Function
是后一个Function
的入参,在函数调用中当实际入参是形参的子类那么是兼容的。
即如下的转换函数式兼容的:
评论