Language/Java
[Java/ERROR] ConcurrentModificationException
강서월
2022. 12. 9. 10:43
상황
주어진 문자열로 재귀적인 방법을 사용해서 집합을 만들기 위해서 다음과 같은 코드를 작성하였다.
public ArrayList<String> makeSet(ArrayList<String> prev, String firstStr, String subStr){
ArrayList<String> result = new ArrayList<>();
if(subStr.equals("")){
result.add("");
result.add(firstStr);
return result;
}
prev = makeSet(prev, String.valueOf(subStr.charAt(0)), subStr.substring(1));
result = prev;
for(String s : result){
result.add(firstStr+s);
}
Collections.sort(result);
return result;
}
firstStr를 제외한 subStr로 만들어진 집합을 ArrayList<String> prev로 할당받고, result에 기존 집합(prev)를 할당하고, result를 순회하면서 firstStr를 추가한 String을 추가하였는데 다음과 같이 ConcurrentModificationException이 발생하였다.
에러코드
원인
이 에러는 허용되지 않았을 때 동시적으로 object를 수정하려고 시도할 때 발생한다고 하며 주로 Java Collection Classes 객체를 순회하면서 요소를 삭제하거나 변경을 할 때 발생한다.
예를 들어, 다음과 같이 list를 순회하면서 value가 3인 값이 있으면 list에서 해당 value를 삭제하는 코드이다.
public class Concurrentmodificationexception {
public static void main(String[] args) {
ArrayList<Integer> list = new ArrayList<>();
list.add(1);
list.add(2);
list.add(3);
list.add(4);
list.add(5);
Iterator<Integer> it = list.iterator();
while (it.hasNext()) {
Integer value = it.next();
System.out.println("List Value:" + value);
if (value.equals(3))
list.remove(value);
}
}
}
iterator로 list를 순회하면서 우리는 이것을 동시적으로 수정하려고 했기 때문에 발생하는 문제이다. Index가 변경되어 일부 요소는 순회하지 않을 수 있다. List에서 문제가 발생할 수 있는 것을 감지하여 ConcurrentModificationException를 발생시킨다.
해결방안
public ArrayList<String> makeSet(ArrayList<String> prev, String firstStr, String subStr){
System.out.println("firstStr = " + firstStr);
System.out.println("subStr = " + subStr);
ArrayList<String> result = new ArrayList<>();
if(subStr.equals("")){
result.add("");
result.add(firstStr);
return result;
}
prev = makeSet(prev, String.valueOf(subStr.charAt(0)), subStr.substring(1));
for(String s : prev){
result.add(s);
result.add(firstStr+s);
}
Collections.sort(result);
return result;
}
다음과 같이 prev를 순회하면서 추가는 result에 하도록 코드를 작성했다.
혹은 크기가 변하지 않는 배열을 사용하거나 synchronize 키워드를 사용하는 방법도 있다.