前言
上一章咱们了解了以下内容:
① 为什么使用lambda表达式
② lambda表达式的语法
③ 函数式接口
这一章咱们继续了解lambda表达式。
1.4 方法引用
咱们先看一下方法引用和非方法引用的区别:
//非方法引用
button.setOnAction(event->Sysout.out.println(event));
//方法引用
button.setOnAction(Sysout.out:println);
正如上面所示:
“ :: ”操作符将方法名和对象或类分隔开来。以下是三种主要情况:
① 对象::实例方法
② 类::静态方法
③ 类::实例方法
在①和②中方法引用等同于方法参数的lambda表达式。如之前所述,Sysout.out::println等同于Sysout.out.println(x)。相似地,Math::pow等同于(x,y)->Math.pow(x,y)。
在③中,第一个参数会成为执行方法的对象。例如:String::compareToIgnoreCase等同于(x,y)->x.compareIgnoreCase(y)。
注意:如果有多个同名的重载方法,那么编译器会从上下文中找到最匹配的一个方法。例如,如果有两个版本的Math.max方法,一个接收整型作为参数,而另一个接收Double类型作为参数,究竟会选择哪个方法,取决于Math.max被转换为函数式接口的方法参数。同lambda表达式类似,方法引用也不会独立存在,它们经常被用于转换为函数式接口的实例。
你还可以捕获方法引用中的this参数。例如,this::equals等同于x->this.equals(x)。当然你也可以使用super对象,例如,super::实例方法
以下用来说明该机制的示例:
class Greeter{
public void greet(){
Sysout.out.println("Hello World");
}
}
class ChildGreeter extends Greeter{
public void greet(){
Thread t=new Thread(super::greet);
t.start();
}
}
当线程启动时,会调用它的Runnable方法,然后执行super::greet并调用父类中的greet方法。
注意:在匿名类中,你可以调用一个闭合类或父类的员工方法,例如闭合类.this::方法或闭合类.super::方法。
1.5 构造器引用
构造器引用同方法引用类似,不同的是构造器引用中方法名是new。例如:Button:new表示Button类的构造器引用。
你可以使用数组类型来编写构造器引用。例如:int[]::new是一个含有一个参数的构造器引用,这个参数就是数组的长度。它等同于lambda表达式x->new int[x]。数组构造器可以用来绕过java中的一个限制。在Java中,无法构造一个泛型类型T的数组。表达式new T[n]是错误的,因为它会被擦除为new Object[n]。这对于编写API的开发人员来说是一个问题。例如,我们希望构造一组按钮,Stream接口中有一个返回Object数组的toArray方法:
Object[] buttons=stream.toArray();
但是这并不能让我们满意。用户希望一组按钮对象而并不是一组Object对象,Stream API通过构造器引用解决了这个问题。它允许将Button[]::new 传给toArray方法:
Button[] buttons=stream.toArray(Button[]::new);
toArray方法会调用该构造器来获得一个正确类型的数组。然后它会填充并返回该数组。