下面的代码是实现同一功能时,使用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);