Java8中方法引用的使用详解

1. 引言

Java8中最受广大开发中喜欢的变化之一是因为引入了 lambda 表达式,因为这些表达式允许我们放弃匿名类,从而大大减少了样板代码,并提高了可读性。
方法引用是lambda表达式的一种特殊类型。它们通常通过引用现有方法来创建简单的lambda表达式。

方法引用包括以下四种类型:

  • 静态方法
  • 特定对象的实例方法
  • 特定类型的任意对象的实例方法
  • 构造方法
  • 在本篇文章中,我们将探讨Java中的方法引用。

2. 引用静态方法

We'll begin with a very simple example, capitalizing and printing a list of Strings:

我们从一个非常简单的示例开始,字符串转成大写并打印:

List<String> messages = Arrays.asList("hello", "baeldung", "readers!");

我们可以通过简单的lambda表达式直接调用 StringUtils.capitalize() 方法:

messages.forEach(word -> StringUtils.capitalize(word));

或者,我们可以使用方法引用来简单地引用 capitalize 静态方法:

messages.forEach(StringUtils::capitalize);

注意,方法引用应使用::运算符。

3. 引用特定对象的实例方法

为了演示这种类型的方法引用,我们新建以下这两个类:

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);

4. 引用特定类型任意对象的实例方法

这种类型的方法引用与前面的示例类似,但不必创建自定义对象来执行比较。

让我们创建一个要排序的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);

尽管它仍然是一行代码,但是方法引用更容易阅读和理解。

5. 引用构造函数

我们可以像在第一个例子中引用静态方法一样引用构造函数。唯一区别是需要使用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构造函数,从而使代码看起来更加简洁明了。

6. 其他示例和限制

目前为止,方法引用是一个使代码非常清晰和易读的好方法。但是,我们不能用它们来代替各种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));

7. 总结

在这篇文章中,我们学习了Java中的方法引用,以及如何使用它们来替换lambda表达式,从而提高了可读性并阐明编程的意图。

到此这篇关于Java 8 中的方法引用使用的文章就介绍到这了,更多相关Java8 方法引用内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Java8方法引用和构造引用代码实例

    这篇文章主要介绍了java8方法引用和构造引用代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 一方法引用概述 方法引用是特定Lamda表达式的一种简写,其思路就是能替换Lamda表达式就直接调用函数使用方法名. 其语法格式:类名 :: 方法名. 二3种方法引用 1 指向静态方法的引用 语法格式: 静态类名(ClassName)::方法名(MethodName) 示例: // 1 Lamda静态方法 @Test public void L

  • java8学习教程之函数引用的使用方法

    前言 我们在上一篇文章 中以实例讲解如何定义和使用 lambda 表达式,以及与其它语言相比,lambda 表达式在 Java 中的特殊规范.并且提到,lambda 表达式可以进一步简化为函数引用. 这篇文章将介绍如何使用函数引用,话不多说了,来一起看看详细的介绍吧. 函数引用的类型 函数引用分为以下四种: 静态函数,比如 Integer 类的 parseInt 函数,可以写作 Integer::parseInt 对象级别函数的引用,比如 String 类的 length 函数,可以写作 Str

  • Java8方法引用及构造方法引用原理实例解析

    如果不熟悉Java8新特性的小伙伴,初次看到函数式接口写出的代码可能会是一种懵逼的状态,我是谁,我在哪,我可能学了假的Java,(・∀・(・∀・(・∀・*),但是语言都是在进步的,就好比面向对象的语言Java也可以写出优雅的函数式调用,学习的过程并不复杂,当你学会了Java8中函数式编程的新特性,你一定会对他爱不释手的.下面介绍一下基于Lambda表达式简写的两种引用.避免再次看到这种代码时的尴尬

  • 30分钟入门Java8之方法引用学习

    前言 之前两篇文章分别介绍了Java8的lambda表达式和默认方法和静态接口方法.今天我们继续学习Java8的新语言特性--方法引用(Method References). 在学习lambda表达式之后,我们通常使用lambda表达式来创建匿名方法.然而,有时候我们仅仅是调用了一个已存在的方法.如下: Arrays.sort(stringsArray,(s1,s2)->s1.compareToIgnoreCase(s2)); 在Java8中,我们可以直接通过方法引用来简写lambda表达式中已

  • Java8新特性之方法引用的实践指南

    一 前言 日常开发中,经常使用到Lambda表达式,例如: public static void main(String[] args) { List<Integer> list = Arrays.asList(1, 5, 10, 4, 2); // 打印列表中的每一个数字 list.forEach((x) -> System.out.println(x)); } 其中(x) -> System.out.println(x)就是使用的Lambda表达式.Lambda表达式可以分为三

  • 如何更好的使用Java8中方法引用详解

    前言 方法引用是用来直接访问类或者实例的已经存在的方法或者构造方法.方法引用提供了一种引用而不执行方法的方式,它需要由兼容的函数式接口构成的目标类型上下文.计算时,方法引用会创建函数式接口的一个实例. 当Lambda表达式中只是执行一个方法调用时,不用Lambda表达式,直接通过方法引用的形式可读性更高一些.方法引用是一种更简洁易懂的Lambda表达式. 注意:方法引用是一个Lambda表达式,其中方法引用的操作符是双冒号"::". 在Java8中,使用方法引用非常简单,如String

  • Java8 Lamda方法引用和构造引用原理

    一方法引用概述 方法引用是特定Lamda表达式的一种简写,其思路就是能替换Lamda表达式就直接调用函数使用方法名. 其语法格式:类名 :: 方法名. 二3种方法引用 1 指向静态方法的引用 语法格式: 静态类名(ClassName)::方法名(MethodName) 示例: // 1 Lamda静态方法 @Test public void LamdaSTest(){ String youku1327 = "1327"; Function function = s -> Obje

  • Java8中如何通过方法引用获取属性名详解

    前言 在我们开发过程中常常有一个需求,就是要知道实体类中Getter方法对应的属性名称(Field Name),例如实体类属性到数据库字段的映射,我们常常是硬编码指定 属性名,这种硬编码有两个缺点. 1.编码效率低:因为要硬编码写属性名,很可能写错,需要非常小心,时间浪费在了不必要的检查上. 2.容易让开发人员踩坑:例如有一天发现实体类中Field Name定义的不够明确,希望换一个Field Name,那么代码所有硬编码的Field Name都要跟着变更,对于未并更的地方,是无法在编译期发现的

  • Java8新特性:Lambda表达式之方法引用详解

    1.方法引用简述 方法引用是用来直接访问类或者实例的已经存在的方法或者构造方法.方法引用提供了一种引用而不执行方法的方式,它需要由兼容的函数式接口构成的目标类型上下文.计算时,方法引用会创建函数式接口的一个实例. 当Lambda表达式中只是执行一个方法调用时,不用Lambda表达式,直接通过方法引用的形式可读性更高一些.方法引用是一种更简洁易懂的Lambda表达式. Lambda表达式全文详情地址:http://blog.csdn.net/sun_promise/article/details/

  • 使用java8的方法引用替换硬编码的示例代码

    背景 想必大家在项目中都有遇到把一个列表的多个字段累加求和的情况,也就是一个列表的总计.有的童鞋问,这个不是给前端做的吗?后端不是只需要把列表返回就行了嘛...没错,我也是这样想的,但是在一场和前端的撕逼大战中败下阵来之后,这个东西就落在我身上了.当时由于工期原因,时间比较紧,也就不考虑效率和易用性了,只是满足当时的需求,就随便写了个方法统计求和.目前稍微闲下来了,就把原来的代码优化下.我们先来看一下原来的代码... 原代码 工具类 import org.apache.commons.lang3

随机推荐