다형성 차이(foreach의 맨 아래 레이어는 Iterator입니다)
Iterator는 인터페이스 유형이므로 컬렉션이나 배열의 유형을 신경 쓰지 않습니다.
for와 Foreach는 먼저 컬렉션의 유형, 심지어 컬렉션에 있는 요소의 유형까지 알아야 합니다.
코드입니다. :
디컴파일된 코드:
먼저 Alibaba Java 개발 매뉴얼
을 살펴보겠습니다. 하지만 1에서는 오류가 보고되지 않으며, 2에서 오류가 보고됩니다(java.util.ConcurrentModificationException)
먼저 ArrayList의 반복자 메소드 구현을 살펴보겠습니다.
는 new Itr()을 호출하여 Itr 클래스(반복자). 이때 Itr의 세 가지 매개변수가 초기화됩니다.
커서는 다음 인덱스 위치를 나타냅니다(0부터 시작).
size는 컬렉션의 크기입니다. (2)
은 예외 클래스
다음 메소드가 검사될 때 확인됩니다. CheckForComodification이 같음
modCount 수정 횟수(추가 및 제거할 때마다 +1됨) ExpectModCount 예상되는 최대 횟수
먼저 삭제 상황을 살펴보겠습니다. "2":
아니요. 하나의 루프:
이때 modCount와 ExpectModCount가 모두 2이기 때문에(modCount는 두 번 추가되었으므로 2입니다) 첫 번째 루프에서는 예외가 발생하지 않으며 예외는 발생하지 않습니다. 보조 사이클의 경우 첫 번째 루프에 발생합니다. 다음 메서드가 완료된 후 foreach 루프 메서드 본문에 있는 제거 메서드의 if 조건이 충족되지 않고 루프가 종료됩니다.
두 번째 루프:
두 번째 루프의 hasNext 및 next 메서드는 성공적으로 완료될 수 있습니다. 그런 다음 foreach 루프 메서드 본문에 제거 메서드를 입력하여 요소를 삭제합니다. 이때 size-1은 1이 됩니다. 제거 메소드의 fastRemove 메소드에 modCount+1이 추가되어 3이 됩니다.
세 번째 루프:
그런 다음 세 번째 루프의 hasNext 메서드로 이동합니다. 정상적인 상황에서는 이 메소드는 false를 반환하지만 이때의 크기가 1로 변경되었고, 이때의 커서가 2(커서는 다음 인덱스 위치를 나타냄)이기 때문에 둘이 같지 않아 에러를 반환한다. true이므로 계속해서 다음 메서드의 checkForComodification 메서드로 이동하여 현재 modCount와 ExpectModCount가 동일한지 확인합니다. 이때 modCount가 예상ModCount 값인 2와 다른 3으로 변경되었기 때문에 여기서 ConcurrentModificationException 예외가 발생했습니다.
1을 삭제할 때 왜 예외가 발생하지 않는지 살펴보겠습니다.
첫 번째 루프:
위와 마찬가지로 이번에는 modCount와 ExpectModCount가 모두 2이므로 첫 번째 루프에서는 둘 다 아닙니다. hasNext 또는 next 메소드는 예외를 발생시킵니다. 그런 다음 foreach 루프 메서드 본문에 제거 메서드를 입력하여 요소를 삭제합니다. 위와 동일하게 size-1은 1이 되고, modCount+1은 3이 됩니다.
두 번째 루프:
두 번째 루프의 hasNext 메소드에서 이때의 커서는 1이고 크기도 1이며 동일합니다. 따라서 hasNext 메서드가 false를 반환하면 foreach 루프에서 벗어나 후속 next 메서드로 이동하지 않으므로 예외가 발생하지 않습니다.
처음
문장 ①은 iterator()를 호출하고,
는 새로운 Itr()을 호출하여 Itr 클래스(iterator)를 생성합니다. 이때 Itr의 세 가지 매개변수가 초기화됩니다.
이때 ExpectModCount == modCount == 2 (리스트는 add 메소드를 동원하기 때문에 add 메소드는 modCount에 ++ 연산을 구현하게 됩니다)
Sentence ②는 아래 hasNext() 메소드를 호출하여 다음의 첨자 커서를 반환합니다. 첫 번째 루프이기 때문에 커서는 0이고 크기는 2입니다(0 != 2 true)
Sentence 3은 foreach 루프 메서드에서 제거 메서드의 if 조건을 호출합니다. 본문이 만족되지 않아 종료됩니다. 이 사이클은
두 번째
문장 ② 아래의 hasNext() 메서드를 호출하고 액세스할 다음 요소의 첨자 커서를 반환합니다.
size는 여전히 2입니다(1 != 2 true)
다음() 메서드는 ③ 문장에서 호출되어 정상적으로 값을 가져오고 첫 번째 요소 "2"를 가져옵니다.
remove() 메서드는 다음과 같습니다. ④ 문장에서 호출되고 해당 요소가 목록에서 성공적으로 삭제됩니다. 참고로, 제거 메소드를 호출하면 modCount++가 있습니다. 이때 모두 modCount3, ExpectModCount2, size1
세 번째
Sentence ②는 아래의 hasNext() 메소드를 호출하여 다음 접근할 요소의 첨자 커서를 반환합니다. 두 번째 루프이므로 커서는 2입니다. , size 1
의 문장 ③에 대해 next() 메서드를 호출합니다. next() 메서드의 첫 번째 문장은 checkForComodification()을 호출하는 것이므로 modCount(3) != ExpectModCount(2)가 발생합니다.
루프가 끝나면 while(iterator.hasNext())는 다음 요소가 있는지 확인하고 완료 후 제거 후 2개를 삭제합니다. , 다음에 입력할 때 커서는 여전히 1이고 크기도 1입니다.
Foreach, Remove2를 삭제한 후 다음에 입력할 때 커서는 2, 크기는 1이므로 false를 반환하고 다음 방법을 사용합니다. 그런 다음 modCount=3 및 예상ModCount=2
iterator 아래의 ArrayList를 보면
위 내용은 Java ArrayList를 순회할 때 요소를 삭제하기 위해 foreach와 iterator를 사용하는 것의 차이점은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!