真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

java中ConcurrentModificationException異常警告怎么解決

這篇文章主要介紹“java中ConcurrentModificationException異常警告怎么解決”,在日常操作中,相信很多人在java中ConcurrentModificationException異常警告怎么解決問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對(duì)大家解答”java中ConcurrentModificationException異常警告怎么解決”的疑惑有所幫助!接下來,請(qǐng)跟著小編一起來學(xué)習(xí)吧!

成都創(chuàng)新互聯(lián)主要為客戶提供服務(wù)項(xiàng)目涵蓋了網(wǎng)頁視覺設(shè)計(jì)、VI標(biāo)志設(shè)計(jì)、網(wǎng)絡(luò)營銷推廣、網(wǎng)站程序開發(fā)、HTML5響應(yīng)式網(wǎng)站建設(shè)公司、手機(jī)網(wǎng)站制作設(shè)計(jì)、微商城、網(wǎng)站托管及網(wǎng)站建設(shè)維護(hù)、WEB系統(tǒng)開發(fā)、域名注冊(cè)、國內(nèi)外服務(wù)器租用、視頻、平面設(shè)計(jì)、SEO優(yōu)化排名。設(shè)計(jì)、前端、后端三個(gè)建站步驟的完善服務(wù)體系。一人跟蹤測試的建站服務(wù)標(biāo)準(zhǔn)。已經(jīng)為成都砂巖浮雕行業(yè)客戶提供了網(wǎng)站制作服務(wù)。

異常分析

相信寫過一些Java代碼的人都遇到過這個(gè)異常,一般都是由以下代碼引起的:

import java.util.List;
import java.util.ArrayList;

public class Test{
   public static void main(String[] args){
     List list = new ArrayList<>();
     list.add("123");
     list.add("456");
     list.add("789");
     for(String obj : list){
         list.remove(obj);
     }
   }
}

上述代碼最終會(huì)引發(fā)java.util.ConcurrentModificationException,那么為什么呢?首先我們將上述代碼反編譯,得到如下結(jié)果(如果對(duì)foreach語法糖比較了解可以忽略):

public class Test {
 public Test();
   Code:
      0: aload_0
      1: invokespecial #1                  // Method java/lang/Object."":()V
      4: return
   LineNumberTable:
     line 4: 0

 public static void main(java.lang.String[]);
   Code:
      0: new           #2                  // class java/util/ArrayList
      3: dup
      4: invokespecial #3                  // Method java/util/ArrayList."":()V
      7: astore_1
      8: aload_1
      9: ldc           #4                  // String 123
     11: invokeinterface #5,  2            // InterfaceMethod java/util/List.add:(Ljava/lang/Object;)Z
     16: pop
     17: aload_1
     18: ldc           #6                  // String 456
     20: invokeinterface #5,  2            // InterfaceMethod java/util/List.add:(Ljava/lang/Object;)Z
     25: pop
     26: aload_1
     27: ldc           #7                  // String 789
     29: invokeinterface #5,  2            // InterfaceMethod java/util/List.add:(Ljava/lang/Object;)Z
     34: pop
     35: aload_1
     36: invokeinterface #8,  1            // InterfaceMethod java/util/List.iterator:()Ljava/util/Iterator;
     41: astore_2
     42: aload_2
     43: invokeinterface #9,  1            // InterfaceMethod java/util/Iterator.hasNext:()Z
     48: ifeq          72
     51: aload_2
     52: invokeinterface #10,  1           // InterfaceMethod java/util/Iterator.next:()Ljava/lang/Object;
     57: checkcast     #11                 // class java/lang/String
     60: astore_3
     61: aload_1
     62: aload_3
     63: invokeinterface #12,  2           // InterfaceMethod java/util/List.remove:(Ljava/lang/Object;)Z
     68: pop
     69: goto          42
     72: return
   LineNumberTable:
     line 6: 0
     line 7: 8
     line 8: 17
     line 9: 26
     line 10: 35
     line 11: 61
     line 12: 69
     line 13: 72
}

將上述代碼翻譯出來等價(jià)于下列代碼:

import java.util.List;
import java.util.ArrayList;
import java.util.Iterator;

public class Test{
   public static void main(String[] args){
     List list = new ArrayList<>();
     list.add("123");
     list.add("456");
     list.add("789");
     Iterator iterator = list.iterator();
     while (iterator.hasNext()){
         String obj = iterator.next();
         list.remove(obj);
     }
   }
}

然后我們查看iterator.hasNext()源碼,可以發(fā)現(xiàn)第一行調(diào)用了checkForComodification方法,我們查看這個(gè)方法:

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

modCount != expectedModCount這個(gè)條件成立的時(shí)候會(huì)拋出ConcurrentModificationException異常,那么這個(gè)條件是怎么成立的呢?

1、首先我們查看modCount的來源,可以發(fā)現(xiàn)modCount的值等于當(dāng)前List的size,當(dāng)調(diào)用List.remove方法的時(shí)候modCount也會(huì)相應(yīng)的減1;

2、然后我們查看expectedModCount的來源,可以看到是在構(gòu)造Iterator(這里使用的是ArrayList的內(nèi)部實(shí)現(xiàn))的時(shí)候,有一個(gè)變量賦值,將modCount 的值賦給了expectedModCount

3、最后當(dāng)我們執(zhí)行循環(huán)調(diào)用List.remove方法的時(shí)候,modCount改變了但是expectedModCount并沒有改變,當(dāng)?shù)谝淮窝h(huán)結(jié)束刪除一個(gè)數(shù)據(jù)準(zhǔn) 備第二次循環(huán)調(diào)用iterator.hasNext()方法的時(shí)候,checkForComodification()方法就會(huì)拋出異常,因?yàn)榇藭r(shí)ListmodCount已經(jīng)變?yōu)?了2,而expectedModCount仍然是3,所以會(huì)拋出ConcurrentModificationException異常;

解決方法

那么如何解決該問題呢?我們查看java.util.ArrayList.Itr(ArrayList中的Iterator實(shí)現(xiàn))的源碼可以發(fā)現(xiàn),在該迭代器中有一個(gè)remove方法可以 刪除當(dāng)前迭代元素,而且會(huì)同時(shí)修改modCountexpectedModCount,這樣在進(jìn)行checkForComodification檢查的時(shí)候就不會(huì)拋出異常了,該remove 方法源碼如下:

public void remove() {
   if (lastRet < 0)
       throw new IllegalStateException();
   checkForComodification();

   try {
       ArrayList.this.remove(lastRet);
       cursor = lastRet;
       lastRet = -1;
       expectedModCount = modCount;
   } catch (IndexOutOfBoundsException ex) {
       throw new ConcurrentModificationException();
   }
}

其中ArrayList.this.remove(lastRet);這一行會(huì)改變modCount的值,而后邊會(huì)同步的修改expectedModCount的值等于modCount的值;

現(xiàn)在修改我們開頭的程序如下就可以正常運(yùn)行了:

import java.util.List;
import java.util.ArrayList;
import java.util.Iterator;

public class Test{
   public static void main(String[] args){
     List list = new ArrayList<>();
     list.add("123");
     list.add("456");
     list.add("789");
     Iterator iterator = list.iterator();
     while (iterator.hasNext()) {
         System.out.println("移除:" + iterator.next());
         iterator.remove();
     }
   }
}

到此,關(guān)于“java中ConcurrentModificationException異常警告怎么解決”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)砀鄬?shí)用的文章!


分享名稱:java中ConcurrentModificationException異常警告怎么解決
新聞來源:http://weahome.cn/article/pcjspp.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部