>  기사  >  Java  >  Java에서 객체의 깊은 복사본(deep clone) 및 얕은 복사본(shallow clone) 직렬화

Java에서 객체의 깊은 복사본(deep clone) 및 얕은 복사본(shallow clone) 직렬화

高洛峰
高洛峰원래의
2016-11-23 10:06:221909검색

1. 얕은 복사와 깊은 복사의 개념

⑴Shallow copy(shallow clone)

복사된 개체의 모든 변수는 원본 개체와 동일한 값을 포함하며, 다른 개체에 대한 모든 참조는 Point로 유지됩니다. 원래 개체에. 즉, 얕은 복사본은 참조하는 개체가 아닌 문제의 개체만 복사합니다.

예:

목록 복제에는 여러 가지 일반적인 방법이 있습니다. 목록 얕은 복사에 대한 몇 가지 일반적인 방법을 나열하겠습니다.

public static void main( String []args){

  List<Map<String,String>> list1 = new ArrayList<Map<String, String>>();

   Map<String,String> map = new HashMap<String, String>();

   map.put("name", "xiaoming");

   map.put("age", "28");

   list1.add(map);

   //克隆方法1:利用原list1作为参数直接构造方法生成。

   List<Map<String,String>> list2 = new ArrayList<Map<String, String>>(list1);

    //克隆方法2:手动遍历将原list1中的元素全部添加到复制表中。

   for(int i = 0, l = list1.size(); i < l; i++)

       list2.add(list1.get(i));

//克隆方法3:调用Collections的静态工具方法 Collections.copy


//克隆方法4:使用System.arraycopy方法进行复制


}

목록 자체는 클래스 유형을 저장할 때 주소 저장만 담당하는 객체입니다. 기본 유형을 저장할 때 저장되는 것은 실제 값입니다. 수천 개의 목록이 있더라도 여전히 몇 가지 요소만 있습니다. 재구성이든, 컬렉션의 복사 방법이든, 시스템의 복사 방법이든, 수동 순회이든 결과는 동일합니다. 이러한 방법은 단순히 이전 요소를 가리키는 몇 가지 주소를 추가하여 ArrayList 개체 자체만 변경합니다. 깊은 복사가 수행되지 않았습니다. (그리고 새로운 새로운 객체의 동작은 전혀 없습니다.) 때로는 원래 메모리의 요소를 사용하는 대신 이러한 요소를 실제로 복사해야 하는 경우도 있습니다. 목록 계층은 이 문제를 구현합니다. 힙 메모리에 묻혀 있는 이러한 데이터를 조작하지 않도록 처음부터 Java 언어를 고려했습니다. 모든 작업은 해당 데이터를 찾을 수 있는 주소로 전달됩니다. 주소 자체가 손실되면 GC에 의해 주소가 삭제됩니다. 그래서 비트 단위로 순회해야 하고, new는 새 객체를 생성하고 원래 값을 할당합니다. 위의 접근 방식이 약간 조정되었다고 느낄 수 있으므로 직렬화된 개체를 사용하여 IO 스트림에서 데이터가 돌아다니도록 만들고 복사할 수 있습니다. 사실, 객체를 스트림으로 직렬화할 때 Java 언어는 실제로 손상되었습니다. 결국 주소만 입력할 수는 없습니다. 게다가 io 흐름은 다른 시스템과 상호 작용해야 합니다. 누군가에게 주소를 보내고 어느 힙에서 찾도록 요청합니까? 따라서 힙 메모리를 새로 할당해야 하는 것은 당연합니다.

⑵딥카피 직렬화(deep clone)

복사된 객체의 모든 변수는 다른 객체를 참조하는 변수를 제외하고 원본 객체와 동일한 값을 포함합니다. 다른 개체를 참조하는 변수는 원래 참조된 개체가 아닌 복사된 새 개체를 가리킵니다. 즉, Deep Copy는 복사 대상 객체가 참조하는 모든 객체를 복사하는 것입니다.

직렬화를 사용하여 Java에서 딥 카피(딥 클론)를 수행합니다. (보다 복잡한 객체의 딥 카피의 clone() 메서드를 다시 작성하지 않으려면 중단점 재개를 구현하도록 프로그래밍할 수도 있습니다.) 등)

객체를 스트림에 쓰는 프로세스는 직렬화 프로세스이지만 Java에서는 이를 "동결" 또는 "피클링" 프로세스라고도 합니다. 스트림에서 객체를 읽는 역직렬화 프로세스입니다. "해동" 또는 "제거" 과정이라고 합니다. 스트림에 기록된 것은 객체의 복사본이고 원본 객체는 여전히 JVM에 존재하므로 "피클에 피클된" 것은 객체의 복사본일 뿐이며 Java 피클은 가능합니다. 아직 신선해요. Java 언어로 객체를 심층 복사하려면 먼저 객체가 직렬화 가능 인터페이스를 구현하도록 만든 다음 객체(실제로 객체의 복사본)를 스트림에 쓴 다음(피클에 피클) 읽을 수 있습니다. 스트림에서 꺼내면(피클을 다시 스트림에 넣음) 객체를 재구성할 수 있습니다. 다음은 소스 코드의 전체 복사본입니다.

public List<Map<String,String>> deClone(Object obj) throws IOException,OptionalDataException,ClassNotFoundException{
//将对象写到流里
ByteArrayOutoutStream bo=new ByteArrayOutputStream();
ObjectOutputStream oo=new ObjectOutputStream(bo);
oo.writeObject(obj);//从流里读出来
ByteArrayInputStream bi=new ByteArrayInputStream(bo.toByteArray());
ObjectInputStream oi=new ObjectInputStream(bi);
return(oi.readObject());
}

이에 대한 전제는 객체와 객체 내에서 참조되는 모든 객체가 직렬화 가능하다는 것입니다. 그렇지 않으면 직렬화할 수 없는 객체나 속성을 임시로 설정할 수 있는지 주의 깊게 조사해야 합니다. 복사 과정에서 제외합니다.


성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.