JAVA 8 '::' 关键字,带你深入了解它!
::
关键字提供了四种语法,可以直接引用已有Java类或对象(实例)的方法或构造器。与lambda
联合使用,::
关键字可以使语言更简洁,减少冗余代码。
语法种类 | 示例 |
---|---|
引用静态方法 | ContainingClass::staticMethodName |
引用特定对象的实例方法 | containingObject::instanceMethodName |
引用特定类型的任意对象的实例方法 | ContainingType::methodName |
引用构造函数 | ClassName::new |
文章目录
- 引用静态方法
- 引用特定对象的实例方法
- 引用特定类型的任意对象的实例方法
- 引用构造函数
引用静态方法
public class Person {
public enum Sex {
MALE, FEMALE
}
String name;
LocalDate birthday;
Sex gender;
String emailAddress;
public int getAge() {
// ...
}
public Calendar getBirthday() {
return birthday;
}
public static int compareByAge(Person a, Person b) {
return a.birthday.compareTo(b.birthday);
}}
假设您的社交网络应用程序的成员包含在一个数组中,并且您想按年龄对数组进行排序。您可以使用以下代码(在示例中找到本节中描述的代码摘录 MethodReferencesTest):
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());
调用的方法签名如下:
static <T> void sort(T[] a, Comparator<? super T> c)
请注意,该接口Comparator
是功能接口。因此,您可以使用lambda
表达式,而不是定义并创建一个新类的实例,该实例实现Comparator
:
Arrays.sort(rosterAsArray,
(Person a, Person b) -> {
return a.getBirthday().compareTo(b.getBirthday());
}
);
但是,这种用于比较两个Person
实例的出生日期的方法已经存在Person.compareByAge
。您可以改为在lambda
表达式的主体中调用此方法:
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)
。每个都有以下特征:
- 它的形参列表是从复制
Comparator<Person>.compare
,这是(Person, Person)
。 - 它的主体调用该方法
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());
}
}
ComparisonProvider myComparisonProvider = 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);
方法参考的等效lambda
表达式String::compareToIgnoreCase
将具有形式参数列表(String a, String b)
,其中a
和b
是用于更好地描述此示例的任意名称。方法引用将调用该方法a.compareToIgnoreCase(b)
。
引用构造函数
您可以使用name
以与静态方法相同的方式引用构造函数new
。以下方法将元素从一个集合复制到另一个:
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
不带任何参数并返回一个对象的方法。因此,您可以transferElements
使用lambda
表达式调用该方法,如下所示:
Set<Person> rosterSetLambda =
transferElements(roster, () -> { return new HashSet<>(); });
您可以使用构造函数引用代替lambda
表达式,如下所示:
Set<Person> rosterSet = transferElements(roster, HashSet::new);
Java
编译器推断您要创建一个HashSet
包含type
元素的集合Person
。或者,您可以指定以下内容:
Set<Person> rosterSet = transferElements(roster, HashSet<Person>::new);
开心一刻
局长与科长共乘电梯,局长放一屁后对科长说:你放屁了!科长说:不是我放的…不久科长被免职,局长在会上说:屁大的事你都担待不起,要你何用?
翻译不易,如果觉得不错的,请帮忙点个赞!!!
还没有评论,来说两句吧...