這篇文章主要介紹“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í)List
的modCount
已經(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í)修改modCount
和expectedModCount
,這樣在進(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í)用的文章!