方法引用通过方法的名字来指向一个方法。
方法引用可以使语言的构造更紧凑简洁,减少冗余代码。
方法引用使用一对冒号 :: 。
下面,我们在 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 特性 —— 方法引用的更多相关文章

  1. JAVA8新特性——方法引用

    JAVA9都要出来了,JAVA8新特性都没搞清楚,是不是有点掉队哦~ 在Lamda新特性的支持下,JAVA8中可以使用lamda表达式来创建匿名方法.然而,有时候我们仅仅是需要调用一个已存在的方法(如 ...

  2. 【Java 8】方法引用

    一.概述 在学习lambda表达式之后,我们通常使用lambda表达式来创建匿名方法.然而,有时候我们仅仅是调用了一个已存在的方法.如下: Arrays.sort(stringsArray,(s1,s ...

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

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

  4. java中的方法引用

    引用静态方法:类名称::static 方法名称: 引用某个对象的方法:对象::普通方法: 引用特定类方法:特定类::方法 引用构造方法:类名称::new 范例:引用静态方法 package com.j ...

  5. Java学习笔记-方法引用

    方法引用(Method Reference) 上一篇中记录了Lambda表达式,其可以创建匿名方法.当Lambda表达式只是调用一个存在的方法时,可以采用方法引用(JDK8具有的特性).如下: pub ...

  6. Java 8Lambda之方法引用(Method References)

    方法引用分为4类,方法引用也受到访问控制权限的限制,可以通过在引用位置是否能够调用被引用方法来判断.具体分类信息如下: 类型 使用方式 静态方法 ContainingClass::staticMeth ...

  7. Java自学-Lambda 方法引用

    Lambda 方法引用 步骤 1 : 引用静态方法 首先为TestLambda添加一个静态方法: public static boolean testHero(Hero h) { return h.h ...

  8. Java基础教程——方法引用

    方法引用 Lambda表达式的代码,是否可以再简洁?--方法引用 对象/类名::方法名 参数都不用写明. import java.util.function.Consumer; public clas ...

  9. java8新特性——方法引用与构造器引用

    上篇文章简单学习了java8内置得4大核心函数式接口,这类接口可以解决我们遇到得大多数得业务场景得问题.今天来简单学习一下方法引用与构造器引用. 一.方法引用 方法引用:若lambda 体中得内容已经 ...

随机推荐

  1. nginx系列13:最少连接算法以及如何跨worker进程生效

    最少连接算法 使用最少连接算法可以使得nginx优先选择连接最少的上游服务器,需要用到upstream_least_conn模块. 如何跨worker进程生效 因为nginx是多进程结构的,默认多个w ...

  2. HeadFirst设计模式读书笔记之策略模式

    1. 例子 1. 做一个鸭子模拟器,里面有很多不同的鸭子,有的可以游泳,有的可以睡觉,有的可以呱呱叫,一般套路是定义一个鸭子的超类,在 超类里定义睡觉,游泳,呱呱叫的方法,再让不同的鸭子子类继承这个超 ...

  3. Hadoop综合大作业

    Hadoop综合大作业 要求: 用Hive对爬虫大作业产生的文本文件(或者英文词频统计下载的英文长篇小说)词频统计. 用Hive对爬虫大作业产生的csv文件进行数据分析 1. 用Hive对爬虫大作业产 ...

  4. 20190421-那些年使用过的CSS预处理器(CSS Preprocessor)之Sass and Less

    写在前面乱七八糟的前言: emmm,还是决定把Sass与Less单独出来写成一篇,可能会稍微好辣么一丢丢?TAT语法特性是真的香,通篇下来能吸收个10%自我感觉已经很nice了,毕竟渣渣的我有渣渣的自 ...

  5. HashTable、ConcurrentHashMap、TreeMap、HashMap关于键值的区别

    集合类 key value super 说明 HashTable 不能为null 不能为null Dictionary 线程安全 ConcurrentHashMap 不能为null 不能为null A ...

  6. 使用 MapTiler 进行地图切片

    在GIS开发中接触比较多的就是切图与发布,通常大家使用的是GlobalMapper.ArcGIS.GDAL等. 一般在使用Leaflet.js或其他框架开发时,使用的是TMS切片格式,大佬们基本用GD ...

  7. OPPO A7x在哪里开启usb调试模式的详细经验

    当我们使用Pc连接安卓手机的时候,如果手机没有开启Usb调试模式,Pc则没法成功读到我们的手机,这时我们需要想办法将手机的Usb调试模式开启,这里我们讲解OPPO A7x如何开启Usb调试模式的方法. ...

  8. VS fopen sprinft ... unsafe 问题

    我的用的是VS2017 VS项目->右键属性(最下面)->C/C++->预处理器->预处理器定义->编辑->加上_CRT_SECURE_NO_WARNINGS  - ...

  9. 【原】无脑操作:IDEA使用时,提示"8080端口被占用"的解决

    1.问题描述:IDEA使用时,提示"8080端口被占用" Description:    The Tomcat connector configured to listen on ...

  10. svn上传*.so文件

    做移动开发,android里面需要用到第三方类库,设计"*.so"文件. svn无法提交,Eclipse里面的svn视图里面该文档无版本图标. 原因描述:svn忽略某些扩展名的文件 ...