写在前面

我们已经知道,lambda表达式是一个匿名函数,可以用lambda表达式来实现一个函数式接口。
 
很自然的,我们会想到类的方法也是函数,本质上和lambda表达式是一样的,那是否也可以用类的方法来实现一个函数式接口呢?答案是可以的。我们称之为方法引用(method reference)。
 
本文的示例代码可从gitee上获取:https://gitee.com/cnmemset/javafp
 

方法引用

一个典型例子,向一个Map中写入单词以及它的长度:
public static void simpleMethodReference() {
Map<String, Integer> wordMap = new HashMap<>(); // 等同于 wordMap.computeIfAbsent("hello", s -> s.length());
wordMap.computeIfAbsent("hello", String::length); // 输出为 {hello=5}
System.out.println(wordMap);
}
上述代码中,String::length 就是方法引用,它用 :: 来分割类名或对象与方法名,:: 左侧是类名或对象,:: 右侧是方法名。
一般来说,方法引用有4种情况:
1. object::instanceMethod —— 对象 + 实例方法
2. Class::staticMethod —— 类名 + 静态方法
3. Class::instanceMethod —— 类名 + 实例方法
4. Class::new —— 类名 + 关键字 new ,这种情况又称为构造器引用(constructor reference)
 

1. object::instanceMethod

object::instanceMethod,:: 左侧是一个对象,:: 右侧是实例方法名。
它等价于提供了 instanceMethod 方法的参数列表的 lambda表达式。
 
形象来说,假设方法 instanceMethod 的参数列表为 (x, y),那么 object::instanceMethod 等价于 (x, y) -> object.instanceMethod(x, y) 。
 
例如对于字符串 str (String str = ""):
str::compareTo 等价于  s -> str.compareTo(s)
 
示例代码如下:
public static void objectInstanceMethodReference() {
String me = "me"; // wordMap 的 key 是给定的单词,value是不区分大小写,与单词 "me" 比较后得出的值
Map<String, Integer> wordMap = new HashMap<>(); // me::compareToIgnoreCase 等价于 s -> me.compareToIgnoreCase(s)
wordMap.computeIfAbsent("him", me::compareToIgnoreCase);
wordMap.computeIfAbsent("you", s -> me.compareToIgnoreCase(s)); System.out.println(wordMap);
}
上述代码的输出是:
{him=5, you=-12}
 

2. Class::staticMethod

Class::staticMethod,:: 左侧是一个类,:: 右侧是静态方法名。
它等价于提供了staticMethod方法的参数列表的lambda表达式。
 
形象来说,假设静态方法 staticMethod 的参数列表为 (x, y),那么 Class::staticMethod 等价于 (x, y, z) -> Class.staticMethod(x, y) 。
 
例如:
System.out::println 等价于 x -> System.out.print(x)
 
示例代码:
public static void classStaticMethodReference() {
List<String> list = Arrays.asList("Guangdong", "Zhejiang", "Jiangsu"); // System.out::println 等价于 s -> System.out.println(s)
list.forEach(System.out::println);
}
上述代码输出为:
Guangdong
Zhejiang
Jiangsu
 

3. Class::instanceMethod

对于Class::instanceMethod,:: 左侧是一个类,:: 右侧是实例方法名。
 
假设 instanceMethod 的参数列表是 (x, y),那么Class::instanceMethod 等价于lambda表达式  (obj, x, y) -> obj.instanceMethod(x, y),其中 obj 是 Class 的对象实例。
 
例如:
String::length 等价于 s -> s.length()
String::compareToIgnoreCase 等价于 (s1, s2) -> s1.compareToIgnoreCase(s2)
 
示例代码:
public static void classInstanceMethodReference() {
Map<String, Integer> wordMap = new HashMap<>();
Integer wordLen = wordMap.computeIfAbsent("hello", String::length);
System.out.println(wordMap);
}
上述代码输出为:
{hello=5}
 

4. Class::new

对于Class::new,new的含义是指Class的构造函数,所以又称为构造器引用(constructor reference)。
 
假设Class的构造函数有两个,它们的参数列表分别是(x)和(x, y),那么 Class::new 可能等价于 x -> new Class(x),也有可能等价于 (x, y) -> new Class(x, y),具体是哪个,编译器会在编译阶段通过上下文推断出来。
 
例如:
BigDecimal::new ,根据上下文,可能等价于 (String s) -> new BigDecimal(s)
 
特别的,数组类型也可以使用构造器引用。数组类型只有一个构造参数,表示数组的长度:
String[]::new 等价于 x -> new String[x]
 
示例代码:
public static void ctorMethodReference() {
List<String> list = Arrays.asList("1.1", "2.2", "3.3"); // BigDecimal::new 根据上下文推断,等价于 s -> new BigDecimal(s)
Stream<BigDecimal> stream = list.stream().map(BigDecimal::new);
List<BigDecimal> decimalList = stream.collect(Collectors.toList());
System.out.println(decimalList); // 构建一个新的 Stream ,之前的 Stream 已经被关闭了
Stream<BigDecimal> stream1 = list.stream().map(BigDecimal::new); // BigDecimal[]::new ,数组的构造器引用,等价于 x -> new BigDecimal[x]
BigDecimal[] decimalArray = stream1.toArray(BigDecimal[]::new);
for (BigDecimal d : decimalArray) {
System.out.println(d);
}
}
上述代码的输出为:
[1.1, 2.2, 3.3]
1.1
2.2
3.3
 

5. this::instanceMethod和super::instanceMethod

对于this::instanceMethod,很容易理解,相当于把this关键字看做是当前类的实例对象即可。
 
例如:
this::equals 等价于 x -> this.equals(x)
 
对于super::instanceMethod,会相对复杂一些,相当于在this对象上,调用的指定方法父类版本。
 
示例代码:
public class SuperMethodReferenceExample {
public static void main(String[] args) {
ThreadWaiter waiter = new ThreadWaiter();
waiter.run();
} public static class Waiter {
public void sayHi() {
System.out.println("Hello, man!");
}
} public static class ThreadWaiter extends Waiter {
@Override
public void sayHi() {
System.out.println("Hello, thread!");
} public void run() {
// 指定调用父类 Waiter 的 sayHi 方法
Thread t = new Thread(super::sayHi);
t.start();
}
}
}

上述代码的输出为:

Hello, man!
 
 

结语

方法引用可以视为lambda表达式的一个语法糖。

Java中的函数式编程(四)方法引用method reference的更多相关文章

  1. Java 中的函数式编程(Functional Programming):Lambda 初识

    Java 8 发布带来的一个主要特性就是对函数式编程的支持. 而 Lambda 表达式就是一个新的并且很重要的一个概念. 它提供了一个简单并且很简洁的编码方式. 首先从几个简单的 Lambda 表达式 ...

  2. Java中的函数式编程(二)函数式接口Functional Interface

    写在前面 前面说过,判断一门语言是否支持函数式编程,一个重要的判断标准就是:它是否将函数看做是"第一等公民(first-class citizens)".函数是"第一等公 ...

  3. 方法引用(Method reference)和invokedynamic指令详细分析

    方法引用(Method reference)和invokedynamic指令详细分析 invokedynamic是jvm指令集里面最复杂的一条.本文将详细分析invokedynamic指令是如何实现方 ...

  4. Java中的函数式编程(三)lambda表达式

    写在前面 lambda表达式是一个匿名函数.在Java 8中,它和函数式接口一起,共同构建了函数式编程的框架.   lambda表达式乍看像是匿名内部类的一种语法糖,但实际上,它们是两种本质不同的事物 ...

  5. Java中的函数式编程(五)Java集合框架中的高阶函数

    写在前面 随着Java 8引入了函数式接口和lambda表达式,Java 8中的集合框架(Java Collections Framework, JCF)也增加相应的接口以适应函数式编程.   本文的 ...

  6. Java中的函数式编程(六)流Stream基础

    写在前面 如果说函数式接口和lambda表达式是Java中函数式编程的基石,那么stream就是在基石上的最富丽堂皇的大厦. 只有熟悉了stream,你才能说熟悉了Java 的函数式编程. 本文主要介 ...

  7. Java中的函数式编程(七)流Stream的Map-Reduce操作

    写在前面 Stream 的 Map-Reduce 操作是Java 函数式编程的精华所在,同时也是最为复杂的部分.但一旦你啃下了这块硬骨头,那你就真正熟悉Java的函数式编程了. 如果你有大数据的编程经 ...

  8. Java中的函数式编程(八)流Stream并行编程

    写在前面 在本系列文章的第一篇,我们提到了函数式编程的优点之一是"易于并发编程". Java作为一个多线程的语言,它通过 Stream 来提供了并发编程的便利性. 题外话: 严格来 ...

  9. java 方法引用(method reference)

    it -> it != null等价于Objects::nonNull

随机推荐

  1. springMVC学习总结(一) --springMVC搭建

    springMVC学习总结(一) --springMVC搭建 搭建项目 1.创建一个web项目,并在项目中的src文件夹下创建一个包com.myl.controller. 2.添加相应jar包 3.在 ...

  2. 【曹工杂谈】Maven底层容器Plexus Container的前世今生,一代芳华终落幕

    Maven底层容器Plexus Container的前世今生,一代芳华终落幕 前言 说实话,我非常地纠结,大家平时只是用Maven,对于内部的实现其实也不关心,我现在非要拉着大家给大家讲.这就有个问题 ...

  3. VUE006. 前端跨域代理服务器ProxyTable概述与配置

    概述 使用  vue-cli  工具生成一个  vue  项目: vue init webpack my-project-vue 在生成的项目结构里,会有一个  index.js  文件.在这个文件里 ...

  4. HCNP Routing&Switching之IS-IS路由渗透和开销

    前文我们了解了IS-IS邻居建立过程.LSDB同步.拓扑计算和路由的形成:回顾请参考https://www.cnblogs.com/qiuhom-1874/p/15265698.html:今天我们来聊 ...

  5. 有备无患「GitHub 热点速览 v.21.38」

    作者:HelloGitHub-小鱼干 数据库最重要的一个功能是容灾备份,备份不只是对数据库重要,对日常工作生活的我们一样重要,比如花了一个工作日写的代码没有备份(虽然可能只有 1 行-)总归是一个让人 ...

  6. 一起学习PHP中断言函数的使用

    原来一直以为断言相关的函数是 PHPUnit 这些单元测试组件提供的,在阅读手册后才发现,这个 assert() 断言函数是 PHP 本身就自带的一个函数.也就是说,我们在代码中进行简单的测试的时候是 ...

  7. Docker DevOps实战:GitLab+Jenkins(2)- CI/CD相关配置

    Jenkins关联GitLab Gitlab仓库配置Webhooks 上传项目到GitLab,Jenkins构建

  8. Jmeter系列(27)- 常用逻辑控制器(6) | 如果(if)控制器If Controller

    如果(if)控制器(If Controller) 在实际工作中,当使用JMeter做性能脚本或者接口脚本时,当遇到需要对不同的条件做不同的操作时,我们可以使用JMeter中if控制器来实现 if控制器 ...

  9. Object of type type is not JSON serializable

    报这个错的原因是因为json.dumps函数发现字典里面有bytes类型的数据,无法编码.解决方法:将bytes类型的数据就把它转化成str类型. 定义dates[]后return JsonRespo ...

  10. 『GoLang』string及其相关操作

    目录 1. 字符串简介 2. 字符串的拼接 3. 有关 string 的常用处理 3.1 strings 包 3.1.1 判断两个 utf-8 编码字符串是否相同 3.1.2 判断字符串 str 是否 ...