Lambda創(chuàng)建一個匿名方法揖赴,這提供了一種比擁有一個方法的匿名類更簡潔的用法错负。如果說Lambda簡單到只需要調(diào)用一個現(xiàn)有的方法韭赘,那么方法引用可以作為一個Lambda表達式來讓代碼變的更加簡潔直觀吁断。
方法是可以引用的
回顧初識Lambda當中介紹的場景男旗,現(xiàn)在管理員需要用戶按照生日進行排序峭状,Lambda可以簡潔直觀的完成這個場景而咆。
List<Person> roster = Person.createRoster();
Person[] rosterAsArray = roster.toArray(new Person[roster.size()]);
Arrays.sort(rosterAsArray,
(a, b) -> {
return a.getBirthday().compareTo(b.getBirthday());
}
);
由于Person
提供了一個靜態(tài)方法compareByAge
摇肌,這個方法簡化排序算法的代碼。
Arrays.sort(rosterAsArray,
(a, b) -> {
return Person.compareByAge(a, b);
}
);
但這并不是最干凈的方式媒役,借助方法引用可以更加簡單。
Arrays.sort(rosterAsArray,Person::compareByAge);
Person::compareByAge
完全等同于(a, b) -> { return Person.compareByAge(a, b); }
宪迟。
因為Lambda表達式僅僅使用到了一個簡單的方法酣衷,所以可以借助方法引用這個特性直接將這個方法作為Lambda表達式使用。
哪類方法可以引用
類型 | 使用方式 |
---|---|
靜態(tài)方法 | ContainingClass::staticMethodName |
指定實例的方法 | containingObject::instanceMethodName |
特定類實例的任意方法 | ContainingType::methodName |
構(gòu)造方法 | ClassName::new |
靜態(tài)方法
正如之前Person::compareByAge
這樣的調(diào)用方式次泽。
指定實例的方法
定義一個各種比較算法的封裝對象穿仪。
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());
}
}
類實例化之后席爽,對象就有兩個可以調(diào)用的比較算法,這個對象的方法就可以用作方法引用啊片。
ComparisonProvider myComparisonProvider = new ComparisonProvider();
Arrays.sort(rosterAsArray, myComparisonProvider::compareByName);
特定類實例的任意方法
這個用法與上面相比稍微特殊一些只锻。
String[] stringArray = { "Barbara", "James", "Mary", "John", "Patricia", "Robert", "Michael", "Linda" };
Arrays.sort(stringArray, String::compareToIgnoreCase);
可以看到sort
操作的對象是String
數(shù)組,compareToIgnoreCase
既不是靜態(tài)方法紫谷,同時也沒有通過使用一個String
對象的方法引用齐饮,而是直接將String
的compareToIgnoreCase
方法作為Lambda。compareToIgnoreCase
實際只有一個輸入?yún)?shù)笤昨,Lambda的目標類型Compartor<T>
的compare(T o1, T o2)
是有兩個輸入?yún)?shù)的祖驱,按照之前的方式,這個引用的方法似乎也是需要兩個String
類型的參數(shù)瞒窒。
其實并非如此捺僻,雖然Lambda有兩個輸入?yún)?shù),這兩個輸入?yún)?shù)可以看做是a
和b
崇裁,方法的引用則與a.compareToIgnoreCase(b)
這種調(diào)用方式相同匕坯。
構(gòu)造方法
定義了一個可以轉(zhuǎn)換集合實現(xiàn)方式的方法,這個方法可以將一種集合對象轉(zhuǎn)換成另外一種集合對象拔稳。
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
是一個JDK標準函數(shù)接口葛峻,它提供了一個get
的無參方法,這里通過get
構(gòu)造了一個Set實例壳炎。如果把一個List集合轉(zhuǎn)換成Set集合泞歉,使用Lambda實現(xiàn):
Set<Person> rosterSetLambda = transferElements(roster, () -> { return new HashSet<>(); });
通過構(gòu)造方法的引用變的更簡潔。
Set<Person> rosterSet = transferElements(roster, HashSet::new);
編譯器可以推斷出HashSet的類型參數(shù)匿辩,也可以顯式聲明類型參數(shù)腰耙。
Set<Person> rosterSet = transferElements(roster, HashSet<Person>::new);