Java 8 特性 —— 方法引用
方法引用通过方法的名字来指向一个方法。
方法引用可以使语言的构造更紧凑简洁,减少冗余代码。
方法引用使用一对冒号 :: 。
下面,我们在 Car 类中定义了 4 个方法作为例子来区分 Java 中 4 种不同方法的引用。
方法引用实际上是某些 Lambda 表达式的更简洁写法,原因就是在这些情况下,编译器能够智能的推断出参数体中的值究竟是方法的传入参数还是调用者。
方法引用有以下四种形式:
| 类型 | 示例 |
| 构造方法引用 | ClassName::new |
| 静态方法引用 | ClassName::staticMethodName |
| 特定类的任意对象的方法引用 | ClassName::methodName |
| 特定对象的实例方法引用 | instance::methodName |
上面的四种形式的实例:
Supplier.java
package java.util.function; @FunctionalInterface
public interface Supplier<T> {
T get();
}
Car.java
package com.boomoom;
import java.util.function.Supplier;
public class Car {
//Supplier是jdk1.8的接口,这里和lambda一起使用了
public static Car create(final Supplier<Car> supplier) {
return supplier.get();
}
public static void collide(final Car car) {
System.out.println("Collided " + car.toString() + " and it went boom!");
}
public void follow(final Car another) {
System.out.println("Following the " + another.toString());
}
public void repair() {
System.out.println("Repaired " + this.toString());
}
}
Maintest.java
package com.boomoom; import java.util.Arrays;
import java.util.List; public class MainTest {
public static void main(String[] args) { // 构造器引用:它的语法是Class::new,或者更一般的Class<T>::new实例如下:
Car[] carList = new Car[2];
final Car car0 = Car.create(Car::new);
carList[0] = car0;
final Car car1 = Car.create(Car::new);
carList[1] = car1;
final List< Car > cars = Arrays.asList( carList ); // 静态方法引用:它的语法是Class::static_method,实例如下:
cars.forEach( Car::collide ); // 特定类的任意对象的方法引用:它的语法是Class::method实例如下:
cars.forEach( Car::repair ); // 特定对象的方法引用:它的语法是instance::method实例如下:
final Car police = Car.create( Car::new );
cars.forEach( police::follow );
}
}
构造方法引用
根据 Supplier 接口的定义,其为函数式接口,我们调用静态方法 create 创建 Car 对象的时候,代码应该是如下所示的:
Car.create(new Supplier<Car>() {
@Override
public Car get() {
return new Car();
}
});
用 Lambda 表达式,更简单的写法就是:
Car.create(()->new Car());
可以看到,Car 类存在一个不带参数的构造方法,所以编译器不需要根据参数列表猜测构造方法的参数(因为都是空的),所以就有一个更加简单的写法:
Car.create(Car::new);
实际上,如果 Lambda 的参数个数和类的构造方法个数一致,也可以改写为上面的形式,只要是没有歧义即可。
静态方法引用
我们创建一个 Car 对象,接着将其添加进一个 List 中:
final Car car = Car.create(Car::new);
final List<Car> cars = Arrays.asList(car);
Java8 中给 Iterable 接口添加了 forEach 方法方便我们遍历集合类型。
现在假设我们要给 List 中的每个 Car 对象调用一次 Car.collide(Car car) 静态方法,那么可以使用 forEach 方法,而 forEach 方法需要传入一个 Consumer,恰好,这个 Consumer接口也带有 FunctionalInterface 注解,所以我们一步一步的来看:
cars.forEach(new Consumer<Car>() {
@Override
public void accept(Car car) {
Car.collide(car);
}
});
写成 Lambda:
cars.forEach(c -> Car.collide(c));
就是对传进来的 Car 对象执行静态方法,很简单。但是实际上,对于静态方法,编译器也不需要推断调用者(类名),当传入参数和静态方法所需参数个数一致时,就不存在歧义:

所以这里可以直接使用方法引用:
cars.forEach(Car::collide);
类成员方法引用
类的成员方法不能是静态的,而这个情况其实和静态方法类似,区别是,Lambda 表达式的参数个数需要等于所调用方法的入参个数加一。
为什么要加一?
因为类的成员方法不能通过类名直接调用,只能通过对象来调用,也就是Lambda表达式的第一个参数,是方法的调用者,从第二个开始的参数个数要和需要调用方法的入参个数一致即可。如下图所示:

对于上面的例子,如果要对List中的每个对象执行一次它的repair方法:
cars.forEach(c -> c.repair());
根据上图,这里参数只有一个,而 repair 方法没有入参,所以不存在歧义,即可以改写为对应的方法引用:
cars.forEach(Car::repair);
对象方法引用
与类方法引用不同的是,对象方法引用方法的调用者是一个外部的对象。如下图:

对于上面例子,可以再创建一个 Car 的对象 police,并让 police 调用 follow 方法跟踪 List 中的每个 Car:
final Car police = Car.create(Car::new);
cars.forEach((car1) -> police.follow(car1));
改成对象方法引用:
cars.forEach(police::follow);
至此,方法引用也完成了。
参考:https://www.cnblogs.com/Fndroid/p/6087380.html
http://www.runoob.com/java/java8-method-references.html
http://www.cnblogs.com/xiaoxi/p/7099667.html
Java 8 特性 —— 方法引用的更多相关文章
- JAVA8新特性——方法引用
JAVA9都要出来了,JAVA8新特性都没搞清楚,是不是有点掉队哦~ 在Lamda新特性的支持下,JAVA8中可以使用lamda表达式来创建匿名方法.然而,有时候我们仅仅是需要调用一个已存在的方法(如 ...
- 【Java 8】方法引用
一.概述 在学习lambda表达式之后,我们通常使用lambda表达式来创建匿名方法.然而,有时候我们仅仅是调用了一个已存在的方法.如下: Arrays.sort(stringsArray,(s1,s ...
- java中的方法引用(method reference)官方文档总结
2017/7/5 转载写明出处:http://www.cnblogs.com/daren-lin/p/java-method-reference.html 今天要说的是java中的一项新特性,方法引用 ...
- java中的方法引用
引用静态方法:类名称::static 方法名称: 引用某个对象的方法:对象::普通方法: 引用特定类方法:特定类::方法 引用构造方法:类名称::new 范例:引用静态方法 package com.j ...
- Java学习笔记-方法引用
方法引用(Method Reference) 上一篇中记录了Lambda表达式,其可以创建匿名方法.当Lambda表达式只是调用一个存在的方法时,可以采用方法引用(JDK8具有的特性).如下: pub ...
- Java 8Lambda之方法引用(Method References)
方法引用分为4类,方法引用也受到访问控制权限的限制,可以通过在引用位置是否能够调用被引用方法来判断.具体分类信息如下: 类型 使用方式 静态方法 ContainingClass::staticMeth ...
- Java自学-Lambda 方法引用
Lambda 方法引用 步骤 1 : 引用静态方法 首先为TestLambda添加一个静态方法: public static boolean testHero(Hero h) { return h.h ...
- Java基础教程——方法引用
方法引用 Lambda表达式的代码,是否可以再简洁?--方法引用 对象/类名::方法名 参数都不用写明. import java.util.function.Consumer; public clas ...
- java8新特性——方法引用与构造器引用
上篇文章简单学习了java8内置得4大核心函数式接口,这类接口可以解决我们遇到得大多数得业务场景得问题.今天来简单学习一下方法引用与构造器引用. 一.方法引用 方法引用:若lambda 体中得内容已经 ...
随机推荐
- Java学习点滴——对象实例化
基于<Java编程思想>第四版 构造与析构 在C++中通过构造函数和析构函数来保证:对象在使用前被正确初始化,在使用后被正确回收.Java中同样存在构造函数,但是没有析构函数.之所以没有析 ...
- 多线程之Thread
Thread类可以创建和控制线程,Thread类的构造函数重载为接受ThreadStart和ParameterizedThreadStart类型的委托参数. Thread类默认创建的是前台线程,所以我 ...
- Python二级-----------程序冲刺1
1. 仅使用 Python 基本语法,即不使用任何模块,编写 Python 程序计算下列数学表达式的结果并输出,小数点后保留3位. ...
- 26 , CSS 构造表单
1. 表单标签使用 2. 下拉菜单背景 3. 滚动条的使用 4. 结构化表单布局 1 1 1 1. . . . 表单标签的使用 <label for=”name”>姓名: <inpu ...
- MFC俄罗斯方块
经典俄罗斯方块游戏 源码百度云链接 链接:https://pan.baidu.com/s/14frk2EuFoiRCzudol2Xgvg提取码:syzk GitHub https://github.c ...
- 测者的测试技术手册:Junit执行单元测试用例成功,mvn test却失败的问题和解决方法
今天遇见了一个奇怪的问题,在IDE中run unit test,全部cases都成功了,但是后来通过mvn test运行case确保错了.在寻求原因的同时也找到了对应的解决方法. Run Unit T ...
- SQL运维
1.碎片扫描 dbcc showconfig('table_name')
- python使用rabbitMQ介绍三(发布订阅模式)
一.模式介绍 在前面的例子中,消息直接发送到queue中. 现在介绍的模式,消息发送到exchange中,消费者把队列绑定到exchange上. 发布-订阅模式是把消息广播到每个消费者,每个消费者接收 ...
- C#几个经常用到的字符串的截取
string str="123abc456";int i=3;1 取字符串的前i个字符 str=str.Substring(0,i); // or str=str.Remov ...
- mssql2008 r2 修改默认端口
1.点击“开始”-“所有程序”-“Microsoft SQL Server 2008R2”-“配置工具”-“SQL Server配置管理器” 2.在打开的“SQL Server配置管理器”窗口中,在左 ...