java8之如何使用函数引用

上一篇文章中以实例讲解如何定义和使用lambda表达式,以及与其它语言相比,lambda表达式在Java中的特殊规范。并且提到,lambda表达式可以进一步简化为函数引用。这篇文章将介绍如何使用函数引用。

函数引用的类型

函数引用分为以下四种:

  • 静态函数,比如Integer类的parseInt函数,可以写作Integer::parseInt
  • 对象级别函数的引用,比如String类的length函数,可以写作String::length
  • 具体实例的函数的引用,比如名称为expensiveTransaction的一个实例的getValue,写作expensiveTransaction::getValue
  • 构造函数的引用

静态函数

比如:

1
Function<String, Integer> stringToInteger = (String s) -> Integer.parseInt(s);

可以写作:

1
Function<String, Integer> stringToInteger = Integer::parseInt;

对象级别函数的引用

1
2
BiPredicate<List<String>, String> contains =
(list, element) -> list.contains(element);

可以写作:

1
BiPredicate<List<String>, String> contains = List::contains;

构造函数的引用

下面再举一个构造函数的例子,首先定义一个函数式接口,唯一的方法get返回指定类型的对象。

1
2
3
4
@FunctionalInterface
public interface Supplier<T> {
T get();
}
1
2
Supplier<TantanitReader> constructor = () -> new TantanitReader();
TantanitReader tantanitReader = constructor.get();

上述代码的lambda表达式new并返回一个新的对象,使得constructor变量成为一个构造器的引用。

等价于以下函数引用:

1
2
Supplier<TantanitReader> constructor2 = TantanitReader::new;
TantanitReader tantanitReader2 = constructor2.get();

以上的例子都是没有参数的情况,下面仍然以构造函数为例,介绍有参数的情况:

1
2
3
public TantanitReader(String loginName) {
this.loginName = loginName;
}
1
2
3
4
5
6
7
Function<String,TantanitReader> constructor3 = (loginName) -> new TantanitReader(loginName);
TantanitReader tantanitReader3 = constructor3.apply("jack");

Function<String,TantanitReader> constructor4 = TantanitReader::new;
TantanitReader tantanitReader4 = constructor4.apply("jack");
TantanitReader tantanitReader5 = constructor4.apply("tom");

这时由于函数只有一个参数,可以使用Java自带的Function<T, R>这个函数式接口,它实际起作用的函数如下:

1
R apply(T t);

作用为根据一个参数,返回一个结果。我们由此可以使用constructor4以及相应的函数引用constructor5。

小结

使用函数引用不仅可以简化lambda表达式,而且在语义上可以更加专注于方法名称,也就是要做的事,抽象级别更加接近人类的认知。所以,应当尽可能使用函数引用。

© 2022 谈谈IT All Rights Reserved. 本站访客数人次 本站总访问量
Theme by hiero