方法引用(Method Reference)
上一篇中记录了Lambda表达式,其可以创建匿名方法。当Lambda表达式只是调用一个存在的方法时,可以采用方法引用(JDK8具有的特性)。如下:
public class Person { public enum Sex {
MALE, FEMALE
} String name;
LocalDate birthday;
Sex gender;
String emailAddress;
int age; public int getAge() {
return age;
} public LocalDate getBirthday() {
return birthday;
} public static int compareByAge(Person a, Person b) {
return a.birthday.compareTo(b.birthday);
}
}
假设需要对一组人员按年龄进行排序,可以采用下边的方式,将人员数组与实现的比较器,传递给Array.sort方法:
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());
当然,可以将PersonAgeComparator的实现采用Lambda表达式,如下:
Arrays.sort(rosterAsArray, (a,b) -> a.birthday.compareTo(b.birthday));
Person类中已经包含根据年龄的比较compareByAge,只需要在Lambda表达式体中直接调用即可:
Arrays.sort( rosterAsArray, (a,b) -> Person.compareByAge(a,b));
由于Lambda表达式调用一个已经存在的方法,可以使用方法引用代替Lambda表达式,如下:
Arrays.sort(rosterAsArray, Person::compareByAge);
其中 ,Person::compareByAge 与 (a, b) -> Person.compareByAge(a, b)是等价的。(1)其参数拷贝于Comparator<Person>.compare,即 (Person, Person);(2)body将调用Person.compareByAge。
方法引用类别
有4种方法引用,如下:
- 引用静态方法,如 ContainingClass::staticMethodName;
- 引用实例方法,如 containingObject::instanceMethodName;
- 引用特殊类型对象的方法,如 ContainingType::methodName;
- 引用构造函数,如 ClassName::new。
(1)引用静态方法
上文中的例子即为静态方法引用。
(2)引用实例方法
即通过类的实例引用方法,如下:
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());
}
}
ComparisonProvider myComparisonProvider = new ComparisonProvider();
Arrays.sort(rosterAsArray, myComparisonProvider::compareByName);
(3)引用特殊类型对象的方法
以String为例:
String[] stringArray = { "Barbara", "James", "Mary", "John",
"Patricia", "Robert", "Michael", "Linda" };
Arrays.sort(stringArray, String::compareToIgnoreCase);
(4)引用构造函数
假设transferElements实现从一个集合到另一个集合拷贝元素,如下:
public static <T, SOURCE extends Collection<T>, DEST extends Collection<T>>
DEST transferElements(
SOURCE sourceCollection,
Supplier<DEST> collectionFactory) { DEST result = collectionFactory.get();
for (T t : sourceCollection) {
result.add(t);
}
return result;
}
其中Supplier包含一个get方法,只返回一个空的集合对象,并且不需要任何参数,可以使用Lambda表达式实现,如下:
Set<Person> rosterSetLambda =
transferElements(roster, () -> { return new HashSet<>(); });
此时可以采用引用构造函数的方法,如下:
Set<Person> rosterSet = transferElements(roster, HashSet::new);
//或
Set<Person> rosterSet = transferElements(roster, HashSet<Person>::new);
总结
- 对于Lambda表达式,当只是调用一个已存在的方法时,可以采用方法引用的方式实现,编译器会自行翻译