使用lambda表达式可以创建简洁的匿名方法。不过,有时候lambda表达式只是简单的调用了已有的方法。此时,使用方法引用无疑是一个更简洁易读的方案。
再来看看之前使用过的Person类:
public class Person { public enum Sex { MALE, FEMALE } String name; LocalDatebirthday; Sexgender; int age; String emailAddress; public int getAge() { return age; } public LocalDategetBirthday() { return birthday; } public static int compareByAge(Person a, Person b) { return a.birthday.compareTo(b.birthday); } }
假设所有的Person对象都保存在一个数组中,然后想按年龄对数组成员进行排序,可以使用如下的代码进行实现:
Person[] rosterAsArray = roster.toArray(new Person[roster.size()]); class PersonAgeComparator implements Comparator<person> { public int compare(Person a, Person b) { return a.getBirthday().compareTo(b.getBirthday()); } } Arrays.sort(rosterAsArray, new PersonAgeComparator());
这里调用的sort方法的声明是这样子的:
static <t> void sort(T[] a, Comparator<? super T> c)
注意这里的Comparator是一个函数式接口,因此无需再定义一个PersonAgeComparator类并创建一个实例,直接使用lambda表达式实现即可:
Arrays.sort(rosterAsArray, (Person a, Person b) -> { return a.getBirthday().compareTo(b.getBirthday()); } );
不过Person类中已经有了一个compareByAge方法,因此可以对上面的表达式作进一步的简化:
Arrays.sort(rosterAsArray, (a, b) -> Person.compareByAge(a, b) );
因为这个lambda表达式只是调用了一个已有的方法,因此可以使用方法引用替换lambda表达式:
Arrays.sort(rosterAsArray, Person::compareByAge);
这里的方法引用Person::compareByAge和lambda表达式(a, b) -> Person.compareByAge(a, b)在语义上是一样的。它们都有如下的特性:
在这个例子以及下面的示例中可以看到,方法引用使用的场合大致是和lambda表达式重合的。
方法引用有四种类型:
类型 |
示例 |
静态方法引用 |
ContainingClass::staticMethodName |
实例方法引用 |
containingObject::instanceMethodName |
一个类任意对象的实例方法引用 |
ContainingType::methodName |
构造方法引用 |
ClassName::new |
前面示例中的Person::compareByAge就是一个静态方法引用。
下面的代码演示了实例方法引用:
class ComparisonProvider { public int compareByName(Person a, Person b) { return a.getName().compareTo(b.getName()); } public int compareByAge(Person a, Person b) { return a.getBirthday().compareTo(b.getBirthday()); } } ComparisonProvidermyComparisonProvider = new ComparisonProvider(); Arrays.sort(rosterAsArray, myComparisonProvider::compareByName);
方法引用myComparisonProvider::compareByName中调用的方法compareByName是对象myComparisonProvider的一部分。JRE会推断出方法的参数类型。在这个例子里就是(Person, Person)。
如下的代码演示了一个类的任意对象的实例方法引用:
String[] stringArray = {"Barbara", "James", "Mary", "John", "Patricia", "Robert", "Michael", "Linda"}; Arrays.sort(stringArray, String::compareToIgnoreCase);
与方法引用String::compareToIgnoreCase对等的lambda表达式需要有(String a, String b)这样的参数列表。这里的a和b只是随意起的名字,只是为了描述参数。方法引用会触发这样的方法:a.compareToIgnoreCase(b)。
可以使用new关键字像创建静态方法引用一样创建构造方法引用。下面的代码将一个集合中的元素拷贝到了另一个集合:
public static <t collection ,><t extends source>, DESTextends Collection<t>> DESTtransferElements(SOURCEsourceCollection, Supplier<dest> collectionFactory) { DESTresult = collectionFactory.get(); for (T t : sourceCollection) { result.add(t); } return result; }
函数式接口Supplier有一个方法get。这个方法没有任何参数,只是返回一个对象:
interface Supplier<t>{ T get(); }
可以使用一个lambda表达式调用 transferElements :
Set<person> rosterSetLambda = transferElements(roster, () -> { return new HashSet<>(); });
也可以使用一个构造方法引用来替换lambda表达式:
Set<person> rosterSet = transferElements(roster, HashSet::new);
java编译器可以推断出你想创建一个HashSet集合,其中包含的元素类型是Person。当然也可以显式声明:
Set<person> rosterSet = transferElements(roster, HashSet<person>::new);
就这样。
https://docs.oracle.com/javase/tutorial/java/javaOO/methodreferences.html
#################