淺談對Java雙冒號::的理解
本文為個人理解,不保證完全正確。官方文檔中將雙冒號的用法分為4類,按照我的個人理解可以分成2類來使用。
官方文檔
官方文檔中將雙冒號的用法分為了以下4類:
用法 舉例 引用靜態(tài)方法 ContainingClass::staticMethodName 引用特定對象的實例方法 containingObject::instanceMethodName 引用特定類型的任意對象的實例方法 ContainingType::methodName 引用構(gòu)造函數(shù) ClassName::new
以下是我的理解
個人理解
雙冒號的作用
在使用雙冒號前我們要先搞清楚一個問題:為什么要使用雙冒號?也就是雙冒號的作用是什么。雙冒號的設(shè)計初衷是為了化簡Lambda表達式,不熟悉Lambda表達式的同學(xué)可以先了解一下。Lambda表達式的形式有兩種:
包含單獨表達式 :parameters -> an expression
list.forEach(item -> System.out.println(item));
包含代碼塊:parameters -> { expressions }
list.forEach(item -> { int numA = item.getNumA(); int numB = item.getNumB(); System.out.println(numA + numB);});
使用雙冒號可以省略第一種Lambda表達式中的參數(shù)部分,即item ->和調(diào)用方法的參數(shù)這兩部分。
例如:
//不使用雙冒號list.forEach(item -> System.out.println(item));//使用雙冒號list.forEach(System.out::println);
雙冒號的使用條件
使用雙冒號有兩個條件:
條件1條件1為必要條件,必須要滿足這個條件才能使用雙冒號。Lambda表達式內(nèi)部只有一條表達式(第一種Lambda表達式),并且這個表達式只是調(diào)用已經(jīng)存在的方法,不做其他的操作。
條件2由于雙冒號是為了省略item ->這一部分,所以條件2是需要滿足不需要寫參數(shù)item也知道如何使用item的情況。有兩種情況可以滿足這個要求,這就是我將雙冒號的使用分為2類的依據(jù)。
情況 舉例 Lambda表達式的參數(shù)與調(diào)用函數(shù)的參數(shù)完全一致 list.forEach(item -> System.out.println(item)) 調(diào)用的函數(shù)是參數(shù)item對象的方法且沒有參數(shù) list.stream().map(item -> item.getId())
一些栗子
Lambda表達式的參數(shù)與調(diào)用函數(shù)的參數(shù)完全一致時
靜態(tài)方法調(diào)用
//化簡前l(fā)ist.forEach(item -> System.out.println(item));//化簡后list.forEach(System.out::println);
非靜態(tài)方法調(diào)用
StringBuilder stringBuilder = new StringBuilder();//化簡前IntStream.range(1, 101).forEach(item -> stringBuilder.append(item));//化簡后IntStream.range(1, 101).forEach(stringBuilder::append);
調(diào)用構(gòu)造方法
官方給出的例子
先定義一個方法,這個方法的作用是將一個集合的內(nèi)容復(fù)制到另一個集合
public <T, SOURCE extends Collection<T>, DEST extends Collection<T>>DEST transferElements(SOURCE sourceCollection, Supplier<DEST> collectionFactory) { DEST result = collectionFactory.get(); result.addAll(sourceCollection); return result;}
調(diào)用這個方法
//化簡前Set<Person> rosterSetLambda = transferElements(roster, () -> new HashSet<>());//化簡后Set<Person> rosterSet = transferElements(roster, HashSet::new);
稍微解釋一下:
調(diào)用時傳入的Lambda表達式相當(dāng)于是對Supplier的繼承,并重寫Supplier的get()方法,下面是Supplier的源碼:
@FunctionalInterfacepublic interface Supplier<T> { /** * Gets a result. * * @return a result */ T get();}
在transferElements()方法中調(diào)用collectionFactory.get()時相當(dāng)于調(diào)用重寫后的方法{return new HashSet<>();}
我自己寫的一個例子
第一個類:
@Datapublic class ModelA { private String id; public ModelA(String id) { this.id = id; } public ModelA() { }}
第二個類
class ClassB { private final List<ModelA> list = new ArrayList<>(); public void add(String string, Function<String, ModelA> function) { list.add(function.apply(string)); }}
測試代碼
ClassB classB = new ClassB();d//化簡前classB.add('ddd', item -> new ModelA(item));//化簡后classB.add('ddd', ModelA::new);
調(diào)用的函數(shù)是參數(shù)item對象的方法且沒有參數(shù)時
//化簡前List<String> stringList = list.stream().map(item -> item.getId()).collect(Collectors.toList());//化簡后List<String> stringList = list.stream().map(ModelA::getId).collect(Collectors.toList());
一種特殊情況
除了上述兩種情況可以使用雙冒號化簡Lambda表達式外,還存在一種特殊情況也可以使用雙冒號。當(dāng)Lambda表達式的參數(shù)有兩個(形如(a,b) -> an expression)時,調(diào)用a的方法參數(shù)為b時,例如:
String[] stringArray = {'Barbara', 'James', 'Mary', 'John'};//化簡前Arrays.sort(stringArray, (a,b) -> a.compareToIgnoreCase(b));//化簡后Arrays.sort(stringArray, String::compareToIgnoreCase);
到此這篇關(guān)于淺談對Java雙冒號::的理解的文章就介紹到這了,更多相關(guān)Java雙冒號::內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!
相關(guān)文章:
1. 在Android中使用WebSocket實現(xiàn)消息通信的方法詳解2. python matplotlib:plt.scatter() 大小和顏色參數(shù)詳解3. Yii2.0引入CSS,JS文件方法4. JSP數(shù)據(jù)交互實現(xiàn)過程解析5. Python importlib動態(tài)導(dǎo)入模塊實現(xiàn)代碼6. vue使用webSocket更新實時天氣的方法7. 淺談python出錯時traceback的解讀8. android studio 打包自動生成版本號與日期,apk輸入路徑詳解9. Nginx+php配置文件及原理解析10. JavaMail 1.4 發(fā)布
