下面的代码是实现同一功能时,使用Lambda表达式与方法引用的对比代码:

1. //使用Lambda表达式对重量排序
2. inventory.sort((Apple a1, Apple a2) -> a1.getWeight().compareTo(a2.getWeight()));
3. //使用方法引用和java.util.Comparator.comparing对重量排序:
4. inventory.sort(comparing(Apple::getWeight));

从对比代码中,方法引用看起来明显要更易读、更自然!

什么是方法引用?

方法引用可以看作是调用特定方法的Lambda的一种快捷写法,如Apple::getWeight可以看作是Lambda表达式(Apple a) -> a.getWeight()的快捷写法。这里要表达的思想是,如果一个Lambda代表的只是“直接调用这个方法”,那最好还是用名称来调用它,而不是去描述如何调用它(Lambda表达式里包含了这种描述)。

方法引用的语法

目标引用::方法的名称

示例(文章开头的代码中):Apple::getWeight

注:不需要括号,Apple::getWeight()是错误的,因为没有实际调用这个方法。

几个Lambda表达式及等效方法引用的例子:

Lambda

等效的方法引用

(Apple a) -> a.getWeight()

Apple::getWeight

() -> Thread.currentThread().dumpStack()

Thread.currentThread()::dumpStack

(str, i) -> str.substring(i)

String::substring

(String s) -> System.out.println(s)

System.out::println

方法引用的分类

方法引用主要有如下三类:

  • (1)指向静态方法的方法引用;
  • (2)指向任意类型实例方法的方法引用;
  • (3)指向现有对象的实例方法的方法引用;
  • (4)构造函数引用

(1)指向静态方法的方法引用

如:Integer的parseInt方法,写作Integer::parseInt。

(2)指向任意类型实例方法的方法引用

如:String的length方法,写作String::length。

(3)指向现有对象的实例方法的方法引用

指向现有对象的方法引用是指在Lambda中调用了一个外部对象的实例方法,如:假设你有一个局部变量expensiveTransaction用于存放Transaction类型的对象,它支持实例方法getValue,那么你就可以写expensiveTransaction::getValue。

(4)构造函数引用

可以用类名加new关键字来创建构造函数的引用:ClassName::new。

默认构造函数

如:

1. //构造函数引用指定默认的Apple()构造函数
2. Supplier<Apple> c1 = Apple::new;
3. //调用Supplier的get方法将产生一个新的Apple
4. Apple a1 = c1.get();

等价于下面的Lambda表达式形式:

1. Supplier<Apple> c1 = () -> new Apple();
2. Apple a1 = c1.get();

带参数的构造函数

假设构造函数的签名是Apple(Integer weight),代码如下:

1. //引用指向Apple(Integer weight)
2. Function<Integer, Apple> c2 = Apple::new;
3. //调用Function的apply方法,产出一个指定重量的苹果
4. Apple a2 = c2.apply(110);

等价于下面的Lambda表达式形式:

1. Function<Integer, Apple> c2 = (weight) -> new Apple(weight);
2. Apple a2 = c2.apply(110);