Java8中最受广大开发中喜欢的变化之一是由于引入了 lambda 表达式,由于这些表达式容许咱们放弃匿名类,从而大大减小了样板代码,并提升了可读性。
方法引用是lambda表达式的一种特殊类型。它们一般经过引用现有方法来建立简单的lambda表达式。java
方法引用包括如下四种类型:编程
在本篇文章中,咱们将探讨Java中的方法引用。api
We'll begin with a very simple example, capitalizing and printing a list of Strings:数组
咱们从一个很是简单的示例开始,字符串转成大写并打印:ide
List<String> messages = Arrays.asList("hello", "baeldung", "readers!");
咱们能够经过简单的lambda表达式直接调用 StringUtils.capitalize()
方法:函数
messages.forEach(word -> StringUtils.capitalize(word));
或者,咱们可使用方法引用来简单地引用 capitalize
静态方法:学习
messages.forEach(StringUtils::capitalize);
注意,方法引用应使用::
运算符。this
为了演示这种类型的方法引用,咱们新建如下这两个类:code
public class Bicycle { private String brand; private Integer frameSize; // standard constructor, getters and setters } public class BicycleComparator implements Comparator { @Override public int compare(Bicycle a, Bicycle b) { return a.getFrameSize().compareTo(b.getFrameSize()); } }
建立一个 BicycleComparator 对象来比较自行车尺寸:对象
BicycleComparator bikeFrameSizeComparator = new BicycleComparator();
咱们可使用lambda表达式按尺寸大小对自行车进行排序,但须要指定两个自行车实例进行比较:
createBicyclesList().stream() .sorted((a, b) -> bikeFrameSizeComparator.compare(a, b));
咱们可使用方法引用让编译器把句柄参数传递给咱们:
createBicyclesList().stream() .sorted(bikeFrameSizeComparator::compare);
这种类型的方法引用与前面的示例相似,但没必要建立自定义对象来执行比较。
让咱们建立一个要排序的Integer 整数列表:
List<Integer> numbers = Arrays.asList(5, 3, 50, 24, 40, 2, 9, 18);
若是咱们使用经典的 lambda 表达式,这两个参数都须要显式传递,而使用方法引用则要简单得多:
numbers.stream() .sorted((a, b) -> a.compareTo(b)); numbers.stream() .sorted(Integer::compareTo);
尽管它仍然是一行代码,可是方法引用更容易阅读和理解。
咱们能够像在第一个例子中引用静态方法同样引用构造函数。惟一区别是须要使用new关键字。
如今咱们用不一样品牌的String列表建立一个Bicycle数组:
List<String> bikeBrands = Arrays.asList("Giant", "Scott", "Trek", "GT");
首先,咱们将向Bicycle类添加一个新的构造函数:
public Bicycle(String brand) { this.brand = brand; this.frameSize = 0; }
接下来,咱们将使用方法引用中的新构造函数,并从原始的String列表中生成一个Bicycle数组:
bikeBrands.stream() .map(Bicycle::new) .toArray(Bicycle[]::new);
注意如何使用方法引用调用Bicycle和Array构造函数,从而使代码看起来更加简洁明了。
目前为止,方法引用是一个使代码很是清晰和易读的好方法。可是,咱们不能用它们来代替各类lambda表达式,由于它们有一些局限性。
它们的主要局限性是因为它们最大的优势:前一个表达式的输出须要与引用的方法声明的输入参数匹配。
看看这个限制的例子:
createBicyclesList().forEach(b -> System.out.printf( "Bike brand is '%s' and frame size is '%d'%n", b.getBrand(), b.getFrameSize()));
这个简单的例子不能用方法引用来表示,由于在咱们的例子中,printf 方法须要3个参数,而使用createBicyclesList().forEach()只容许方法引用一个参数(Bicycle对象)。
最后,咱们研究下,如何建立一个能够从lambda表达式引用的no-operation函数。
在本例中,咱们但愿使用lambda表达式而不使用其参数。
首先,建立 doNothingAtAll 方法:
private static <T> void doNothingAtAll(Object... o) { }
由于这是一个varargs方法,它可执行在任意 lambda 表达式中,而无论引用的对象或参数的数量。咱们看看它的做用:
createBicyclesList() .forEach((o) -> MethodReferenceExamples.doNothingAtAll(o));
在这篇文章中,咱们学习了Java中的方法引用,以及如何使用它们来替换lambda表达式,从而提升了可读性并阐明编程的意图。
若是你以为文章还不错,记得关注公众号: 锅外的大佬
锅外的大佬博客