Java中如何克隆集合——ArrayList和HashSet深拷贝

  编程人员经常误用各个集合类提供的拷贝构造函数作为克隆ListSetArrayListHashSet或者其他集合实现的方法。需要记住的是,Java集合的拷贝构造函数只提供浅拷贝而不是深拷贝,这意味着存储在原始List和克隆List中的对象是相同的,指向Java堆内存中相同的位置。增加了这个误解的原因之一是对于不可变对象集合的浅克隆。由于不可变性,即使两个集合指向相同的对象是可以的。字符串池包含的字符串就是这种情况,更改一个不会影响到另一个。使用ArrayList的拷贝构造函数创建雇员List的拷贝时就会出现问题,Employee类不是不可变的。在这种情况下,如果原始集合修改了雇员信息,这个变化也将反映到克隆集合。同样如果克隆集合雇员信息发生变化,原始集合也会被更改。绝大多数情况下,这种变化不是我们所希望的,克隆对象应该与原始对象独立。解决这个问题的方法是深克隆集合,深克隆将递归克隆对象直到基本数据类型或者不可变类。本文将了解一下深拷贝ArrayList或者HashSet等集合类的一种方法。如果你了解深拷贝与浅拷贝之间的区别,那么理解集合深克隆的方法就会很简单。

Java集合的深克隆

下面例子有一个Employee集合,Employee是可变对象,成员变量namedesignation。它们存储在HashSet中。使用java.util.Collection接口的addAll()方法创建集合拷贝。然后修改存储在原始集合每个Employee对象的designation值。理想情况下这个改变不会影响克隆集合,因为克隆集合和原始集合应该相互独立,但是克隆集合也被改变了。修正这个问题的方法是对存储在Collection类中的元素深克隆。

 1 /**
 2  *
 3  * @ClassName: CollectionCloningTest
 4  * TODO
 5  * @author xingle
 6  * @date 2015-3-20 下午3:32:22
 7  */
 8 public class CollectionCloningTest {
 9
10     public static void main(String[] args){
11         ArrayList<Employee> org = new ArrayList<Employee>();
12         org.add(new Employee("Joe", "Manager"));
13         org.add(new Employee("Tim", "Developer"));
14         org.add(new Employee("Frank", "Developer"));
15
16         Collection<Employee> copy = new HashSet<>(org);
17
18         System.out.println("原来的集合: "+org);
19         System.out.println("复制的集合: "+copy);
20
21         Iterator<Employee> orgItr = org.iterator();
22         while(orgItr.hasNext()){
23             orgItr.next().setDesignation("staff");
24
25         }
26
27         System.out.println("修改后原来的集合: "+org);
28         System.out.println("修改后复制的集合: "+copy);
29     }
30
31 }
32
33
34 class Employee {
35     private String name;
36     private String designation;
37
38     public Employee(String name, String designation) {
39         this.name = name;
40         this.designation = designation;
41     }
42
43     public String getDesignation() {
44         return designation;
45     }
46
47     public void setDesignation(String designation) {
48         this.designation = designation;
49     }
50
51     public String getName() {
52         return name;
53     }
54
55     public void setName(String name) {
56         this.name = name;
57     }
58
59     @Override
60     public String toString() {
61         return String.format("%s: %s", name, designation );
62     }
63
64 }

执行结果:

可以看到改变原始CollectionEmployee对象(改变designation为”staff“)在克隆集合中也有所反映,因为克隆是浅拷贝,指向堆中相同的Employee对象。为了修正这个问题,需要遍历集合,深克隆Employee对象,在这之前,要重写Employee对象的clone方法。

1)Employee实现Cloneable接口
2)为Employee类增加下面的clone()方法

3)不使用拷贝构造函数,使用下面的代码来深拷贝集合

 1 public class CollectionCloningTest {
 2
 3     public static void main(String[] args){
 4         ArrayList<Employee> org = new ArrayList<Employee>();
 5         org.add(new Employee("Joe", "Manager"));
 6         org.add(new Employee("Tim", "Developer"));
 7         org.add(new Employee("Frank", "Developer"));
 8
 9        //Collection<Employee> copy = new HashSet<>(org);
10        Collection<Employee> copy = new HashSet<Employee>(org.size());
11
12
13         System.out.println("原来的集合: "+org);
14         System.out.println("复制的集合: "+copy);
15
16         Iterator<Employee> orgItr = org.iterator();
17         while(orgItr.hasNext()){
18             //orgItr.next().setDesignation("staff");
19             copy.add(orgItr.next().clone());
20
21         }
22
23
24         Iterator<Employee> orgItr2 = org.iterator();
25         while(orgItr2.hasNext()){
26             orgItr2.next().setDesignation("staff");
27         }
28         System.out.println("修改后原来的集合: "+org);
29         System.out.println("修改后复制的集合: "+copy);
30     }
31
32 }
33
34
35 class Employee implements Cloneable{
36     private String name;
37     private String designation;
38
39     public Employee(String name, String designation) {
40         this.name = name;
41         this.designation = designation;
42     }
43
44     public String getDesignation() {
45         return designation;
46     }
47
48     public void setDesignation(String designation) {
49         this.designation = designation;
50     }
51
52     public String getName() {
53         return name;
54     }
55
56     public void setName(String name) {
57         this.name = name;
58     }
59
60     @Override
61     public String toString() {
62         return String.format("%s: %s", name, designation );
63     }
64
65     @Override
66     protected Employee clone(){
67         try {
68             Employee result = (Employee) super.clone();
69             return result;
70         } catch (CloneNotSupportedException e) {
71              throw new RuntimeException(e); // won‘t happen
72         }
73
74     }
75 }

执行结果:

可以看到克隆集合和原始集合相互独立,它们指向不同的对象。

这就是Java中如何克隆集合的内容。现在我们知道拷贝构造函数或者ListSet等各种集合类的addAll()方法仅仅创建了集合的浅拷贝,而且原始集合和克隆集合指向相同的对象。为避免这个问题,应该深克隆集合,遍历集合克隆每个元素。尽管这要求集合中的对象必须支持深克隆操作。

时间: 2024-10-15 17:37:46

Java中如何克隆集合——ArrayList和HashSet深拷贝的相关文章

java中数组、集合、字符串之间的转换,以及用加强for循环遍历

java中数组.集合.字符串之间的转换,以及用加强for循环遍历: 1 @Test 2 public void testDemo5() { 3 ArrayList<String> list = new ArrayList<String>(); 4 list.add("甲乙1"); 5 list.add("甲乙2"); 6 list.add("甲乙3"); 7 list.add("甲乙4"); 8 //

JAVA中所有与集合有关的实现类都是这六个接口的实现类

JAVA中所有与集合有关的实现类都是这六个接口的实现类. Collection接口:集合中每一个元素为一个对象,这个接口将这些对象组织在一起,形成一维结构. List接口代表按照元素一定的相关顺序来组织(在这个序列中顺序是主要的),List接口中数据可重复. Set接口是数学中集合的概念:其元素无序,且不可重复.(正好与List对应) SortedSet会按照数字将元素排列,为"可排序集合". Map接口中每一个元素不是一个对象,而是一个键对象和值对象组成的键值对(Key-Value)

Java中常用的集合

  有序列 允许元素重复否 Collection 否 是 List 是 是 Set AbstractSet 否 否 HashSet TreeSet 是(用二叉树排序) Map AbstractMap 否 使用key-value来映射和存储数据, Key必须惟一,value可以重复 HashMap TreeMap 是(用二叉树排序) 几个面试常见问题:1.Q:ArrayList和Vector有什么区别?HashMap和HashTable有什么区别?A:Vector和HashTable是线程同步的(

Java中数组和集合容器的剖析

java中常用的存储容器就是数组的集合,每种容器存储的形式和结构又有所不同. 数组,是最基础的容器,在创建数组的时候有三种方式分别如下: int[] arr = new int[5]; int[] arr = new String[]{1,2,3,4,5}; int[] arr = {1,2,3,4,5}; 从上面的三种方式可以看出,在定义数组的时候有个共同的特点就是能够直接看出数组的长度,这也是数组的一大特点,就是定义的时候指定长度,同时数组一旦定义完成后长度就不可以变化,这也是数组在后期开发

[转]Java中常用的集合—初学者的你不可错过的精编整理

集合一直都是项目中非常常见的,我是一个Android开发者,集合对于我来说,在项目中使用的次数非常之多,因为使用的多,熟能生巧,所以这里呢!就给那些初学者整理一下Java当中常用的集合吧!   因为此篇文章是给初学者看到,所以对于集合的认识,我们就不从内存的角度去分析了,等你Java学到一定的时候,再去学习一下集合的底层实现,这会让成为一名更加牛的Java程序员.   在整理之前呢,我们先聊一聊为什么集合会这么常用?,集合这个概念,我们初次接触是在高中的数学当中,高中的集合具有以下知识点:  1

Java中泛型在集合框架中的应用

泛型是Java中的一个重要概念,上一篇文章我们说过,当元素存入集合时,集合会将元素转换为Object类型存储,当取出时也是按照Object取出的,所以用get方法取出时,我们会进行强制类型转换,并且通过代码也可以看出来,我们放入其他类型时,如字符串,编译器不会报错,但是运行程序时会抛出类型错误异常,这样给开发带来很多不方便,用泛型就解决了这个麻烦 泛型规定了某个集合只能存放特定类型的属性,当添加类型与规定不一致时,编译器会直接报错,可以清楚的看到错误 当我们从List中取出元素时直接取出即可,不

Java中list&lt;Object&gt;集合去重实例

一:Java中list去重的方法很多,下面说一下其中一种方法:把list里的对象遍历一遍,用list.contain(),如果不存在就放入到另外一个list集合中: 二:实例 这里需要注意的是:使用contains方法的时候,list中里面的对象是否相等的问题,我们知道对象是否相等,有两层意思,对象的地址相等和对象的属性值相等.而contains比对的时候调用的是object类中的equals方法: 我们可以看到,比对的是对象的地址.而实际中可能我们想要的结果是,对象里面的值想等,我们就认为这两

Java中在类定义时对HashSet的初始化方法

Java中很多时候都要用到HashSet的查找功能,那么在类的定义时,数据成员假如就是HashSet类型的,我们定义数据成员之后,不好直接调用add函数来实现初始化,这个时候怎么办? 我们可以这样来做: public static final HashSet<String> salesWords = new HashSet<String>() {{ add("销售额"); add("销售"); add("销售金额"); a

Java中常见的集合框架

1. 一.collection (有序)接口的实现的接口 set  list 其中set接口的实现类是HashSet,List接口的实现类是ArrayList.LinkList.Vector 二.Map(无序)接口的实现类是HashMap.HashTable ArrayList和Vector都是使用数组方式存储数据,此数组元素数大于实际存储的数据以便增加和插入元素,它们都允许直接按序号索引元素,但是插入元素要涉及数组元素移动等内存操作,所以索引数据快而插入数据慢,Vector由于使用了synch