Java에서 깊은 복사(deep copy)로 리스트를 복사하는 방법을 소개합니다.
깊은 복사(deep copy)와 얕은 복사(shallow copy)의 차이점은, 얕은 복사로 리스트를 복사하였을 때는 기존 리스트의 요소를 변경했을 때 복사본의 요소도 변경됩니다. 원본의 변경에도 복사본이 영향을 받지 않으려면 깊은 복사를 하여 객체 전체를 복사해야합니다.
1. 클래스에 clone()을 구현하여 깊은 복사
리스트에 저장되는 요소의 클래스에 clone()
이 구현되어 있으면, clone()
을 이용하여 깊은 복사를 할 수 있습니다.
클래스에 clone()
메소드를 구현하려면, 아래와 같이 Cloneable
을 구현하고 clone()
함수를 오버라이드하면 됩니다.
clone()
메소드에서는 super.clone()
으로 객체의 복제본을 만들 수 있으며, 필요에 따라서 본제된 객체에 일부 변수를 직접 생성하여 할당할 수도 있습니다.
public static class Student implements Cloneable {
public String name;
public int age;
public Student(String name, int age) {
this.name = name;
this.age = age;
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
@Override
public String toString() {
return "Student { name: " + name + ", age: " + age + " }";
}
}
아래 예제는 Student 리스트를 깊은 복사하고, 원본 리스트의 요소를 변경되었을 때 복사된 리스트의 요소도 함께 변경되는지 확인하는 예제입니다. 결과를 보면, 예상한대로 원본 리스트의 요소만 변경되었습니다.
import java.util.ArrayList;
import java.util.List;
public class Example {
public static void main(String[] args) {
List<Student> origList = new ArrayList<>();
origList.add(new Student("John", 20));
origList.add(new Student("Doe", 30));
List<Student> deepCopyList = new ArrayList<>();
for (Student student : origList) {
deepCopyList.add((Student) student.clone());
}
Student first = origList.get(0);
first.name = "Son";
first.age = 32;
System.out.println("origList: " + origList);
System.out.println("deepCopyList: " + deepCopyList);
}
public static class Student implements Cloneable {
public String name;
public int age;
public Student(String name, int age) {
this.name = name;
this.age = age;
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
@Override
public String toString() {
return "Student { name: " + name + ", age: " + age + " }";
}
}
}
Output:
origList: [Student { name: Son, age: 32 }, Student { name: Doe, age: 30 }]
deepCopyList: [Student { name: John, age: 20 }, Student { name: Doe, age: 30 }]
2. 얕은 복사(shallow copy)와 깊은 복사(deep copy)의 차이점
얕은 복사는 원본 리스트에 있는 요소를 복사되는 리스트에 단순히 추가만 하는 것을 말합니다. 즉, 리스트의 객체는 다르지만 원본 리스트와 복사본 리스트에 저장되는 요소는 동일한 객체가 됩니다.
객체를 깊게 복사하지 않기 때문에 성능과 비용적인 측면에서 이득이 될 수 있는데, 원본 리스트의 요소가 변경될 때 복사본의 요소도 함께 변경됩니다.
아래 예제는 얕은 복사로 리스트를 복사하고, 원본 요소를 변경했을 때 변경되는지 확인하는 예제입니다. 결과를 보면 함께 변경되었습니다.
import java.util.ArrayList;
import java.util.List;
public class Example1 {
public static void main(String[] args) throws CloneNotSupportedException {
List<Student> origList = new ArrayList<>();
origList.add(new Student("John", 20));
origList.add(new Student("Doe", 30));
List<Student> shallowCopyList = new ArrayList<>();
for (Student student : origList) {
shallowCopyList.add(student);
}
Student first = origList.get(0);
first.name = "Son";
first.age = 32;
System.out.println("origList: " + origList);
System.out.println("shallowCopyList: " + shallowCopyList);
}
public static class Student implements Cloneable {
public String name;
public int age;
public Student(String name, int age) {
this.name = name;
this.age = age;
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
@Override
public String toString() {
return "Student { name: " + name + ", age: " + age + " }";
}
}
}
Output:
origList: [Student { name: Son, age: 32 }, Student { name: Doe, age: 30 }]
shallowCopyList: [Student { name: Son, age: 32 }, Student { name: Doe, age: 30 }]
Related Posts
- Java - Unsupported class file major version 61 에러
- Java - String.matches()로 문자열 패턴 확인 및 다양한 예제 소개
- Java - 문자열 공백제거 (trim, replace)
- Java - replace()와 replaceAll()의 차이점
- Java - ArrayList 초기화, 4가지 방법
- Java - 배열 정렬(Sorting) (오름차순, 내림차순)
- Java - 문자열(String)을 비교하는 방법 (==, equals, compare)
- Java - StringBuilder 사용 방법, 예제
- Java - 로그 출력, 파일 저장 방법 (Logger 라이브러리)
- Java IllegalArgumentException 의미, 발생 이유
- Java - NullPointerException 원인, 해결 방법
- Seleninum의 ConnectionFailedException: Unable to establish websocket connection 해결
- Java - compareTo(), 객체 크기 비교
- Java - BufferedWriter로 파일 쓰기
- Java - BufferedReader로 파일 읽기
- Java charAt() 함수 알아보기
- Java - BigInteger 범위, 비교, 연산, 형변환
- Java contains()로 문자(대소문자 X) 포함 확인
- Java - Set(HashSet)를 배열로 변환
- Java - 문자열 첫번째 문자, 마지막 문자 확인
- Java - 문자열 한글자씩 자르기
- Java - 문자열 단어 개수 가져오기
- Java - 1초마다 반복 실행
- Java - 배열을 Set(HashSet)로 변환
- Java - 여러 Set(HashSet) 합치기
- Java - 명령행 인자 입력 받기
- Java - 리스트 역순으로 순회, 3가지 방법
- Java - 특정 조건으로 리스트 필터링, 3가지 방법
- Java - HashMap 모든 요소들의 합계, 평균 계산
- Java - 특정 조건으로 HashMap 필터링
- Java - 싱글톤(Singleton) 패턴 구현
- Java - 숫자 왼쪽에 0으로 채우기
- Java - String 배열 초기화 방법
- Java - 정렬된 순서로 Map(HashMap) 순회
- Java - HashMap에서 key, value 가져오기