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. js try catch 获取错误信息

    try{ alert(i); }catch(e){ console.log(e.message,e.name,e.lineNumber) } message -- 错误提示信息 fileName -- ...

  2. 如何用css设置鼠标属性的小手

    在元素中添加onmouseover="this.style.cursor='hand'"或者 cursor:pointer;

  3. Spider--实战--selenium_12306

    # login12306_02 # 图像识别涉及到深度学习,这里直接将验证码识别任务发送到大佬的验证码解析地址,不过现在已经失效了,程序跑到这会报错. # 用户名和密码存储在本地工作目录中的 user ...

  4. Integer a=1,b=1,c=500,d=500;a==b,c==d;

    public class test { public static void main(String[] args){ Integer a=1,b=1,c=500,d=500; System.out. ...

  5. PF_PACKET抓包mmap

    PACKET套接口创建 内核函数packet_create处理PF_PACKET套接口的创建工作.其参数sock->type决定了采用哪一种工作模式,如果参数type为SOCK_PACKET即第 ...

  6. dp背包 面试题 08.11. 硬币

    https://leetcode-cn.com/problems/coin-lcci/ 硬币.给定数量不限的硬币,币值为25分.10分.5分和1分,编写代码计算n分有几种表示法.(结果可能会很大,你需 ...

  7. 基于FFmpeg的Dxva2硬解码及Direct3D显示(四)

    初始化硬解码上下文 目录 初始化硬解码上下文 创建解码数据缓冲区 创建IDirectXVideoDecoder视频解码器 设置硬解码上下文 解码回调函数 创建解码数据缓冲区 这一步为了得到 LPDIR ...

  8. UNP第13章——守护进程

    1. 守护进程的启动方法 (1)系统初始化脚本启动,在系统启动阶段,按照如/etc目录或/etc/rc开头的目录中的某些脚本启动,这些守护进程一开始就有超级用户权限.如inetd,cron,Web服务 ...

  9. 通过ceph-deploy安装不同版本ceph

    之前有在论坛写了怎么用 yum 安装 ceph,但是看到ceph社区的群里还是有人经常用 ceph-deploy 进行安装,然后会出现各种不可控的情况,虽然不建议用ceph-deploy安装,但是既然 ...

  10. hmac检验客户端合法性

    1.服务端 # 验证客户端是否合法 # 不依靠登陆认证 # 当有一个客户端访问你的时候,建立了tcp后,server端主动发起一个数据,数据为想加密的bytes类型的数据. # 客户端拿到要加密的by ...