(注:本文章参看自《Java 8 实战》厄马(Raoul-Gabriel Urma) / 弗斯科(Mario Fusco) / 米克罗夫特(Alan Mycroft))
java8中,设计师在java.util.function包中引入了几个新的函数式接口:Predicate 、 Consumer 和 Function。
1.1 Predicate
java.util.function.Predicate<T>接口定义了一个名叫test的抽象方法,它接受泛型T对象,并返回一个boolean。这恰恰和你先前创建的一样,现在就可以直接使用了。在你需要表示一个涉及类型T的布尔表达式时,就可以使用这个接口。比如,你可以定义一个接受String对象的Lambda表达式,如下所示。
@FunctionalInterface
public interface Predicate<T>{
boolean test(T t);
}
示例:
public class Demo {
public static void main(String[] args) {
List<String> listOfStrings = new ArrayList<>();
listOfStrings.add("hello");
listOfStrings.add("world.");
listOfStrings.add("");
Predicate<String> nonEmptyStringPredicate = (String s) -> !s.isEmpty();
List<String> nonEmpty = filter(listOfStrings, nonEmptyStringPredicate);
System.out.println(nonEmpty);
// List<String> nonEmpty = listOfStrings.stream()
// .filter(item -> !item.isEmpty())
// .collect(Collectors.toList());
}
public static <T> List<T> filter(List<T> list, Predicate<T> p) {
List<T> results = new ArrayList<>();
for (T s : list) {
if (p.test(s)) {
results.add(s);
}
}
return results;
}
}
2.1 Consumer
java.util.function.Consumer<T>定义了一个名叫accept的抽象方法,它接受泛型T的对象,没有返回(void)。你如果需要访问类型T的对象,并对其执行某些操作,就可以使用这个接口。比如,你可以用它来创建一个forEach方法,接受一个Integers的列表,并对其中每个元素执行操作。在下面的代码中,你就可以使用这个forEach方法,并配合Lambda来打印列表中的所有元素。
@FunctionalInterface
public interface Consumer<T>{
void accept(T t);
}
示例:
public class Demo {
public static void main(String[] args) {
List<String> listOfStrings = new ArrayList<>();
listOfStrings.add("hello");
listOfStrings.add("world.");
listOfStrings.add("");
forEach(listOfStrings, System.out::println);
}
public static <T> void forEach(List<T> list, Consumer<T> c) {
for (T i : list) {
c.accept(i);
}
}
}
其中,Lambda是Consumer中accept方法的实现。
3.1 Function
java.util.function.Function<T, R>接口定义了一个叫作apply的方法,它接受一个 泛型T的对象,并返回一个泛型R的对象。如果你需要定义一个Lambda,将输入对象的信息映射到输出,就可以使用这个接口(比如提取苹果的重量,或把字符串映射为它的长度)。在下面的代码中,我们向你展示如何利用它来创建一个map方法,以将一个String列表映射到包含每个 String长度的Integer列表。
@FunctionalInterface
public interface Function<T, R>{
R apply(T t);
}
示例:
public class Demo {
public static void main(String[] args) {
List<String> listOfStrings = new ArrayList<>();
listOfStrings.add("hello");
listOfStrings.add("world");
listOfStrings.add("!!");
System.out.println(map(listOfStrings, String::length));
}
public static <T, R> List<R> map(List<T> list,
Function<T, R> f) {
List<R> result = new ArrayList<>();
for (T s : list) {
result.add(f.apply(s));
}
return result;
}
}
其中,Lambda是Function接口的apply方法的实现。
(未完待续。。)