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

注意如何使用方法引用调用BicycleArray构造函数,从而使代码看起来更加简洁明了。

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 中的方法引用,轻松减少代码量,提升可读性!的更多相关文章

  1. Java 8 中的方法引用

    一.原理概要 lambda 表示式,可以作为某些匿名内部类的替代.主要目的是调用该内部类中的方法,而该方法的实现(重写)由 lambda表示式决定. 通常,我们可能不关心匿名内部类中的具体方法(被重写 ...

  2. Java8中的[方法引用]“双冒号”——走进Java Lambda(四)

    前面的章节我们提及到过双冒号运算符,双冒号运算就是Java中的[方法引用],[方法引用]的格式是 类名::方法名 注意是方法名哦,后面没有括号“()”哒.为啥不要括号,因为这样的是式子并不代表一定会调 ...

  3. Swift编程语言中的方法引用

    由于Apple官方的<The Swift Programming Guide>对Swift编程语言中的方法引用介绍得不多,所以这里将更深入.详细地介绍Swift中的方法引用. Swift与 ...

  4. thymeleaf模板引擎调用java类中的方法(附源码)

    前言 <Docker+SpringBoot+Mybatis+thymeleaf的Java博客系统开源啦> 由于开源了项目的缘故,很多使用了My Blog项目的朋友遇到问题也都会联系我去解决 ...

  5. Jsp中如何通过Jsp调用Java类中的方法

    Jsp中如何通过Jsp调用Java类中的方法 1.新建一个项目,在src文件夹下添加一个包:如:cn.tianaoweb.com; 2.再在包中添加一个类:如 package com; public ...

  6. 前端程序员的蜕变——JS的 event 对象属性、使用实例、兼容性处理(极大提高代码效率、减少代码量)

    下面讨论一下 js 中的 Event 对象,主要从以下三个方面详细的描述(点击标题可跳转到对应部分): 1.什么是event 2.怎么用event,用他该注意什么,几个简单实际应用 3.event在不 ...

  7. 想减少代码量,快设置一个有感知的 Aware Spring Bean

    摘要:正常情况下,Spring 中的 Bean 对 Spring 是无感知的,Spring 框架提供了这种扩展能力,能让一个 bean 成为有感知的. 本文分享自华为云社区<有感知的 Aware ...

  8. WPF INotifyPropertyChanged 通过特性减少代码量

    在很多地方需要用上INotifyPropertyChanged的接口,MVVM模式,List等集合都会用到. 通常我们使用 protected void OnChange(PropertyChange ...

  9. java中的方法引用(method reference)官方文档总结

    2017/7/5 转载写明出处:http://www.cnblogs.com/daren-lin/p/java-method-reference.html 今天要说的是java中的一项新特性,方法引用 ...

随机推荐

  1. Java8 新特性 —— Stream 流式编程

    本文部分摘自 On Java 8 流概述 集合优化了对象的存储,大多数情况下,我们将对象存储在集合是为了处理他们.使用流可以帮助我们处理对象,无需迭代集合中的元素,即可直接提取和操作元素,并添加了很多 ...

  2. 企业网络拓扑RSTP功能实例

    组网图形  RSTP简介 以太网交换网络中为了进行链路备份,提高网络可靠性,通常会使用冗余链路.但是使用冗余链路会在交换网络上产生环路,引发广播风暴以及MAC地址表不稳定等故障现象,从而导致用户通信质 ...

  3. linux netfilter nat1

    linux netfilter nat1 2020整理云笔记上传

  4. Lamport面包店算法详解(转 侵删)

    范例1: boolean  choosing[n];表示进程是否在取号 int  number[n];记录每个进程取到的号码 这些数据结构分别初始化为false和0,为了方便,定义如下符号: 若a&l ...

  5. python之《线程与进程》

    多线程的应用场景 不适用cpu操作密集型任务, 适合io操作密集型任务 同一进程中的数据是互通的,因为python多线程是假多线程,我们要用到多核就需要开多个进程来实现,但是坏处是数据不能互通 线程: ...

  6. ceph扩展bluestore的db分区

    前言 在ceph 14版本里面才加入了bluefs-bdev-migrate,分区迁移相关的命令,那么在12版本里面其实也是可以扩展分区的 测试的版本 [root@lab102 ceph-0]# ce ...

  7. ipmi常用的命令行命令

    前言 记录一些常用的命令行操作 命令 查询机器的电源状态 ipmitool -I lanplus -U admin -P admin -H 172.16.21.215 power status 硬重启 ...

  8. Monitor的扩展支持string的超时锁

    对Monitor的使用可以防止lock的时间过长并且可以设置其对应的超时时间达到对预期代码的一个控制,合理的使用timeout可以有助于程序的健壮性.但是对于不同的并发程序可能某些时候我们需要的粒度是 ...

  9. LTMU论文解析

    LTMU 第零部分:前景提要 一般来说,单目标跟踪任务可以从以下三个角度解读: A matching/correspondence problem.把其视为前后两帧物体匹配的任务(而不考虑在跟踪过程中 ...

  10. 这份java多线程笔记,你真得好好看看,我还没见过总结的这么全面的

    1.线程,进程和多线程 1.程序:指指令和数据的有序集合,其本身没有任何意义,是一个静态的概念 2.进程:指执行程序的一次执行过程,是一个动态的概念.是系统资源分配的单位(注意:很多多线程是模拟出来的 ...