有些时候,你想要传递给其他代码的操作已经有实现的方法了。示例:
button.setOnAction(event -> System.out.println(event);
如果你能够只将println方法传递给setOnAction方法,就更好了!下面是改后的代码:
button.setOnAction(System.out::println);
表达式System.out::println是一个方法引用,等同于lambda表达式:
x -> System.out.println(x)
正如示例代码所示:操作符将方法名和对象或类的名字分开来。以下三种主要的使用情况:
- 对象::实例方法
- 类::静态方法
- 类::实例方法
在前两种情况中,方法引用等同于提供方法参数的lambda表达式。如之前所述,System.out::println等同于 System.out.println(x)。相似的,Math::pow等同于(x,y) -> Math.pow(x,y)。在第三种情况中,第一个参数会成为执行方法的对象。例如String::compareToIgnoreCase等同于(x,y) -> x.compareToIgnoreCase(y)。
注意:如果有多个同名的重载方法,编译器会试图从上下文中找到最匹配的一个方法。例如,有两个版本的Math.max方法,一个接收整型作为参数,而另一个接收double类型的值。究竟会选择哪一个方法,取决于Math::max被转换为的函数式接口的方法参数。同lambda表达式类似,方法引用也不会独立存在,它们经常被用于转换为函数式接口的实例。
你还可以捕获方法引用中的this参数。例如,this::equals就等同于x -> this.equals(x)。你也可以使用super对象,如:super::实例方法。示例:
package java8; public class J2 { public static void main(String[] args) { Greeter g = new ConcurrentGreeter(); g.greet(); } } class Greeter { public void greet(){ System.out.println("Hello world!"); } } class ConcurrentGreeter extends Greeter{ @Override public void greet() { //super代表当前类的对象的父类对象,并不是函数式接口的父实例对象 Thread t = new Thread(super::greet); t.start(); } }
package java8; public class J3 { public static void main(String[] args) { Greeter1 g = new Greeter1(); g.greet(); } } class Greeter1 { public void greet() { //this代表的当前类的对象,并不是函数式接口的实例对象 Thread t = new Thread(this::printInfo); t.start(); } public void printInfo(){ System.out.println("test success!!"); } }
时间: 2024-10-07 09:48:03