标签:not code demo 代码执行 分析 lam ++ boolean end
前面一篇文章介绍了两个常用函数式接口Supplier
和Consumer
接口,今天接着讲另外另个重要函数式接口Predicate
和Function
接口。这几个接口在Java中应用广泛,比方Stream流式API中就经常用到。
当我们需要对某种数据类型进行判断,得到一个boolean值结果时候,可以采用java.util.function.Predicate<T>
接口。比方说对传进来的List判断是否包含某个特定字符PHP。
我们有时候需要进行业务判断,假设我要实现一个功能入参为List,如果List参数满足某个条件则输出对应的结果。比方说下面的代码checkchar()
如果入参list1
满足某个条件那么输出System.out.println("满足对应条件")
,具体条件是什么还不确定,可能是判断List中是否包含某个元素,大小是否符合要求。具体由调用方来实现。
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->{return s.contains("PHP")}
来进行判断。这意思就是说在调用checkChar
方法的时候如果list1
列表中包含PHP
这个元素执行System.out.println("满足对应的条件")
语句。
上述代码执行结果为:
满足对应的条件
list1中包含PHP 字符
上面我们说了单个条件判断,如果想多个条件判断怎么办?可以使用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++字符
"与"、"或"我们已经了解了,剩下还有一个"非"(取反)操作,我们看一下取反操作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#
java.util.function.Function<T,R>
相当于数据中的函数,一个类型的数据作为输入得到另一个类型数据的输出。
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);
}
如果我们想做多步转换那么就需要用到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<String,Integer> one
入参为String
返回值为Integer
,那么后面Function two
的入参必须是Integer
。即前一个Function
的返回值跟后一个Function
入参类型应该兼容。
前面我们说了,即前一个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<? super R, ? extends V> 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);
//...
}
标签:not code demo 代码执行 分析 lam ++ boolean end
原文地址:https://blog.51cto.com/14820287/2506129