欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java中的forEach循環(huán)詳細(xì)解讀

 更新時(shí)間:2023年12月19日 09:11:10   作者:初念初戀  
這篇文章主要介紹了Java中的forEach循環(huán)詳細(xì)解讀,不要再foreach循環(huán)里面進(jìn)行元素的add和remove,如果你非要進(jìn)行remove元素,那么請(qǐng)使用Iterator方式,如果存在并發(fā),那么你一定要選擇加鎖,需要的朋友可以參考下

前言

相信大家肯定都看過阿里巴巴開發(fā)手冊(cè),而在阿里巴巴開發(fā)手冊(cè)中明確的指出,不要再foreach循環(huán)里面進(jìn)行元素的add和remove,如果你非要進(jìn)行remove元素,那么請(qǐng)使用Iterator方式,如果存在并發(fā),那么你一定要選擇加鎖。

foreach

    public static void main(String[] args) {
        List<String> list = new ArrayList<String>();
        list.add("11");
        list.add("22");
        list.add("33");
        list.add("44");
        for (String s : list) {
            if ("22".equalsIgnoreCase(s)) {
                list.remove(s);
            }
        }
        System.out.println(JSONObject.toJSONString(list));
    }

輸出結(jié)果:

Exception in thread "main" java.util.ConcurrentModificationException
    at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:901)
    at java.util.ArrayList$Itr.next(ArrayList.java:851)
    at org.example.list.Test01.main(Test01.java:22)
Process finished with exit code 1

分析異常:

final void checkForComodification() {
    if (modCount != expectedModCount)
        throw new ConcurrentModificationException();
}

比較兩個(gè)值 modCount 和expectedModCount,那么這兩個(gè)變量是什么呢?

其中modCount表示集合的修改次數(shù),這其中包括了調(diào)用集合本身的add方法等修改方法時(shí)進(jìn)行的修改和調(diào)用集合迭代器的修改方法進(jìn)行的修改。而expectedModCount則是表示迭代器對(duì)集合進(jìn)行修改的次數(shù)。

先來看看反編譯之后的代碼,如下:

    public static void main(String[] args) {
        List<String> list = new ArrayList();
        list.add("11");
        list.add("22");
        list.add("33");
        list.add("44");
        Iterator var2 = list.iterator();
        while(var2.hasNext()) {
            String s = (String)var2.next();
            if ("22".equalsIgnoreCase(s)) {
                list.remove(s);
            }
        }
        System.out.println(JSONObject.toJSONString(list));
    }

看里面使用的也是迭代器,也就是說,其實(shí) foreach 每次循環(huán)都調(diào)用了一次iterator的next()方法, foreach方式中調(diào)用的remove方法,是ArrayList內(nèi)部的remove方法,會(huì)更新modCount屬性

我們可以看看ArrayList類中的remove方法

    public E remove(int index) {
        rangeCheck(index);
        modCount++;
        E oldValue = elementData(index);
        int numMoved = size - index - 1;
        if (numMoved > 0)
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);
        elementData[--size] = null; // clear to let GC do its work
        return oldValue;
    }

看到此方法中,有一個(gè)modCount++的操作,也就是說,modCount會(huì)一直更新變化。

我們第一次迭代的時(shí)候 11 != 22 ,直接迭代第二次,這時(shí)候就相等了,執(zhí)行remove()方法,這時(shí)候就是modCount++,再次調(diào)用next()的時(shí)候,modCount = expectedModCount 這個(gè)就不成立了,所以異常信息出現(xiàn)了,其實(shí)也可以理解為在 hasNext() 里面,cursor != size 而這時(shí)候就會(huì)出現(xiàn)錯(cuò)誤了。

也就是說 remove方法它只修改了modCount,并沒有對(duì)expectedModCount做任何操作。

迭代器

為什么阿里巴巴的規(guī)范手冊(cè)會(huì)這樣子定義?

img

它為什么推薦我們使用 Iterator呢?

直接使用迭代器會(huì)修改expectedModCount,而我們使用foreach的時(shí)候,remove方法它只修改了modCount,并沒有對(duì)expectedModCount做任何操作,而Iterator就不會(huì)這個(gè)樣子。

   public static void main(String[] args) {
        List<String> list = new ArrayList<String>();
        list.add("11");
        list.add("22");
        list.add("33");
        list.add("44");
        Iterator<String> iterator = list.iterator();
        while (iterator.hasNext()){
            String item = iterator.next();
            if("22".equals(item)){
                iterator.remove();
            }
        }
        System.out.println(JSONObject.toJSONString(list));
    }

輸出結(jié)果:

["11","33","44"]
Process finished with exit code 0

可以看出結(jié)果是正確的,下面我們來分析一下:

先來看看反編譯之后的代碼:

    public static void main(String[] args) {
        List<String> list = new ArrayList();
        list.add("11");
        list.add("22");
        list.add("33");
        list.add("44");
        Iterator iterator = list.iterator();
        while(iterator.hasNext()) {
            String item = (String)iterator.next();
            if ("22".equals(item)) {
                iterator.remove();
            }
        }
        System.out.println(JSONObject.toJSONString(list));
    }

主要觀察remove()方法的實(shí)現(xiàn),那么需要先看 ArrayList.class:

    public Iterator<E> iterator() {
        return new Itr();
    }
    /**
     * An optimized version of AbstractList.Itr
     */
    private class Itr implements Iterator<E> {
        int cursor;       // index of next element to return
        int lastRet = -1; // index of last element returned; -1 if no such
        int expectedModCount = modCount;
        Itr() {}
        public boolean hasNext() {
            return cursor != size;
        }
        public void remove() {
            if (lastRet < 0)
                throw new IllegalStateException();
            checkForComodification();     //第一步
            try {
                ArrayList.this.remove(lastRet);   //第二步:調(diào)用list的remove方法
                cursor = lastRet;
                lastRet = -1;
                expectedModCount = modCount; 		//第三步:modCount是remove方法去維護(hù)更新,
                                                    //由于第一步中校驗(yàn) modCount 和 expectedModCount 是否相當(dāng)?shù)?
            } catch (IndexOutOfBoundsException ex) {
                throw new ConcurrentModificationException();
            }
        }
        final void checkForComodification() {
            if (modCount != expectedModCount)
                throw new ConcurrentModificationException();
        }
    }
  1. 調(diào)用 checkForComodification()方法,作用:判斷modCount 和 expectedModCount 是否相當(dāng);
  2. foreach 方式中調(diào)用的remove方法,是ArrayList內(nèi)部的remove方法,會(huì)更新modCount屬性;
  3. 將更新后的modCount重新賦值給expectedModCount變量。

Java8的新特性

    public static void main(String[] args) {
        List<String> list = new ArrayList<String>();
        list.add("11");
        list.add("22");
        list.add("33");
        list.add("44");
        list.removeIf("22"::equals);
        System.out.println(JSONObject.toJSONString(list));
    }

總結(jié)

for-each循環(huán)不僅適用于遍歷集合和數(shù)組,而且能讓你遍歷任何實(shí)現(xiàn)Iterator接口的對(duì)象;最最關(guān)鍵的是它還沒有性能損失。

而對(duì)數(shù)組或集合進(jìn)行修改(添加刪除操作),就要用迭代器循環(huán)。所以循環(huán)遍歷所有數(shù)據(jù)的時(shí)候,能用它的時(shí)候還是選擇它吧。

到此這篇關(guān)于Java中的forEach循環(huán)詳細(xì)解讀的文章就介紹到這了,更多相關(guān)forEach循環(huán)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 使用mockito編寫測(cè)試用例教程

    使用mockito編寫測(cè)試用例教程

    這篇文章主要為大家介紹了使用mockito編寫測(cè)試用例教程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-08-08
  • SpringCloud feign服務(wù)熔斷下的異常處理操作

    SpringCloud feign服務(wù)熔斷下的異常處理操作

    這篇文章主要介紹了SpringCloud feign服務(wù)熔斷下的異常處理操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-06-06
  • springboot:接收date類型的參數(shù)方式

    springboot:接收date類型的參數(shù)方式

    這篇文章主要介紹了springboot:接收date類型的參數(shù)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • Java Http多次請(qǐng)求復(fù)用同一連接示例詳解

    Java Http多次請(qǐng)求復(fù)用同一連接示例詳解

    這篇文章主要為大家介紹了Java Http多次請(qǐng)求復(fù)用同一連接示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-10-10
  • java實(shí)現(xiàn)圖書管理系統(tǒng)

    java實(shí)現(xiàn)圖書管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)圖書管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-03-03
  • springboot 如何取消starter的自動(dòng)注入

    springboot 如何取消starter的自動(dòng)注入

    這篇文章主要介紹了springboot 如何取消starter的自動(dòng)注入操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-09-09
  • 在springboot中如何集成clickhouse進(jìn)行讀寫操作

    在springboot中如何集成clickhouse進(jìn)行讀寫操作

    本文介紹了在Spring Boot中集成ClickHouse的步驟,包括引入依賴、配置數(shù)據(jù)源、編寫實(shí)體類和Mapper類進(jìn)行CRUD操作,特別提到批量插入時(shí)需要在SQL語句中添加`FORMAT`以避免錯(cuò)誤,在實(shí)際應(yīng)用中,與MySQL的操作類似,只需將ClickHouse當(dāng)作MySQL使用
    2024-11-11
  • Java多線程atomic包介紹及使用方法

    Java多線程atomic包介紹及使用方法

    這篇文章主要介紹了Java多線程atomic包介紹及使用方法,涉及原子更新基本類型介紹及代碼示例,具有一定參考價(jià)值,需要的朋友可以了解下。
    2017-11-11
  • springboot validator枚舉值校驗(yàn)功能實(shí)現(xiàn)

    springboot validator枚舉值校驗(yàn)功能實(shí)現(xiàn)

    這篇文章主要介紹了springboot validator枚舉值校驗(yàn)功能實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-01-01
  • java實(shí)現(xiàn)MD5加密方法匯總

    java實(shí)現(xiàn)MD5加密方法匯總

    本文給大家匯總介紹了2種java實(shí)現(xiàn)MD5加密的方法,非常的實(shí)用,這里分享給大家,學(xué)習(xí)下其中的思路,對(duì)大家學(xué)習(xí)java非常有幫助。
    2015-10-10

最新評(píng)論