Java for-each循環(huán)使用難題2例(高級使用方法)
Java中,for-each循環(huán)簡化了任何Collection或array的遍歷過程,但并不是每個Java程序員都了解本文將要描述的for-each 循環(huán)的一些細節(jié)。與 Java5 發(fā)布的其他術(shù)語:釋放別名泛型,自動封裝和可變參數(shù)不同,Java開發(fā)者對for-each循環(huán)的使用比任何其他特性更加頻繁,但當問及高級的for-each循環(huán)怎樣工作,或什么是在for-each循環(huán)中使用Collection時的基本需求時,就不是每個人都能夠回答的了。
本篇教程和例子旨在通過深入研究for-each 循環(huán)中幾個有趣的難題來填補上述空白(說明上述問題)。好了,不再贅述,一起看看我們在Java5 for-each循環(huán)的第一個問題。
高級循環(huán)問題 1
考慮下面這段遍歷一個用戶自定義的aggregator或collection類的代碼,這段代碼將會打印出什么,拋出異常還是編譯器錯誤:
package test;
/**
* Java Class to show how for-each loop works in Java
*/
public class ForEachTest {
public static void main(String args[]){
CustomCollection<String> myCollection = new CustomCollection<String>();
myCollection.add("Java");
myCollection.add("Scala");
myCollection.add("Groovy");
//What does this code will do, print language, throw exception or compile time error
for(String language: myCollection){
System.out.println(language);
}
}
}
下面是我們的CustomCollection類,這是個參數(shù)為泛型的類,與任何其他的Collection類相似,依靠于ArrayList并提供從Collection中添加和刪除項的方法。
package test;
public class CustomCollection<T>{
private ArrayList<T> bucket;
public CustomCollection(){
bucket = new ArrayList();
}
public int size() {
return bucket.size();
}
public boolean isEmpty() {
return bucket.isEmpty();
}
public boolean contains(T o) {
return bucket.contains(o);
}
public boolean add(T e) {
return bucket.add(e);
}
public boolean remove(T o) {
return bucket.remove(o);
}
}
答案:
上述代碼將無法通過編譯,這是因為我們的CustomCollection類沒有實現(xiàn)java.lang.Iterable接口,編譯期錯誤如下:
Exception in thread "main" java.lang.RuntimeException: Uncompilable source code - for-each not applicable to expression type
required: array or java.lang.Iterable
found: test.CustomCollection
at test.ForEachTest.main(ForEachTest.java:24)
從中了解到的一個有趣的事實是:for-each循環(huán)僅應(yīng)用于實現(xiàn)了Iterable接口的Java array和Collection類,而且既然所有內(nèi)置Collection類都實現(xiàn)了java.util.Collection接口,已經(jīng)繼承了Iterable,這一細節(jié)通常會被忽略,這點可以在Collection接口的類型聲明“ public interface Collection extends Iterable”中看到。所以為了解決上述問題,你可以選擇簡單地讓CustomCollection實現(xiàn)Collection接口或者繼承AbstractCollection,這是默認的通用實現(xiàn)并展示了如何同時使用抽象類和接口以獲取更好的靈活性?,F(xiàn)在讓我們來看看for-each循環(huán)的第二個難題:
Java for-each循環(huán)的第二個難題:
在下面的代碼示例將會拋出ConcurrentModificationException異常。這里我們使用標準iterator和for-each循環(huán)遍歷ArrayList,隨后刪除元素,你需要找出哪段代碼將會拋出ConcurrentModificationException ,為什么?請注意,答案可能是兩個都會,都不會或其中之一。
package test;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
/**
* Java class to demonstrate inner working of for-each loop in Java
* @author Javin Paul
**/
public class ForEachTest2 {
public static void main(String args[]){
Collection<String> list = new ArrayList<String>();
list.add("Android");
list.add("iPhone");
list.add("Windows Mobile");
// Which Code will throw ConcurrentModificationException, both,
// none or one of them
// example 1
Iterator<String> itr = list.iterator();
while(itr.hasNext()){
String lang = itr.next();
list.remove(lang);
}
// example 2
for(String language: list){
list.remove(language);
}
}
}
大約70%的Java開發(fā)者都會說第一個代碼塊會拋出ConcurrentModificationException異常,因為我們沒有用iterator的remove方法來刪除元素,而是使用ArrayList的 remove()方法。但是,沒有多少Java開發(fā)者會說出for-each循環(huán)也會出現(xiàn)同樣的問題,因為我們在這里沒有使用iterator。事實上,第二個代碼片段也會拋出ConcurrentModificationException異常,這點在解決了第一個困惑之后就變得很明顯了。既然for-each循環(huán)內(nèi)部使用了Iterator來遍歷Collection,它也調(diào)用了Iterator.next(),這會檢查(元素的)變化并拋出ConcurrentModificationException。你可以從下面的輸出中了解到這點,在注釋掉第一個代碼段后,當你運行第二個代碼段時會得到下面的輸出。
Exception in thread "main" java.util.ConcurrentModificationException
at java.util.AbstractList$Itr.checkForComodification(AbstractList.java:372)
at java.util.AbstractList$Itr.next(AbstractList.java:343)
at test.ForEachTest2.main(ForEachTest2.java:34)
以上就是關(guān)于Java5 for-each循環(huán)的全部內(nèi)容。我們已經(jīng)看到了Java程序員在編寫遍歷Collection類的代碼時產(chǎn)生的很多問題,特別是在遍歷collection的同時刪除元素的時候。請牢記,在從任何Collection(例如Map、Set或List)中刪除對象時總要使用Iterator的remove方法,也請謹記for-each循環(huán)只是標準Iterator代碼標準用法之上的一種語法糖(syntactic sugar)而已。
譯者注:語法糖(syntactic sugar),也譯為糖衣語法,是由英國計算機科學(xué)家彼得·約翰·蘭達-Peter J. Landin發(fā)明的一個術(shù)語,指計算機語言中添加的某種語法,這種語法對語言的功能并沒有影響,但是更方便程序員使用。通常來說使用語法糖能夠增加程序的可讀性,從而減少程序代碼出錯的機會。
相關(guān)文章
SpringBoot+Redis實現(xiàn)查找附近用戶的示例代碼
SpringDataRedis提供了十分簡單的地理位置定位的功能,本文主要介紹了SpringBoot+Redis實現(xiàn)查找附近用戶的示例代碼,具有一定的參考價值,感興趣的可以了解一下2024-02-02如何巧用HashMap一行代碼統(tǒng)計單詞出現(xiàn)次數(shù)詳解
這篇文章主要給大家介紹了關(guān)于如何巧用HashMap一行代碼統(tǒng)計單詞出現(xiàn)次數(shù)的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習或者工作具有一定的參考學(xué)習價值,需要的朋友們下面來一起學(xué)習學(xué)習吧2020-07-07Springboot整合redis實現(xiàn)發(fā)布訂閱功能介紹步驟
發(fā)布訂閱作為一種設(shè)計思想在很多開源組件中都有體現(xiàn),比如大家熟知的消息中間件等,可謂把發(fā)布訂閱這一思想體現(xiàn)的淋漓盡致了2022-09-09