在下面的例子中,我们有一个可变Employee对象的集合,每个对象包含name和designation字段,将它们保存在HashSet中。我们用java.util.Collection接口中的addAll()方法来创建这个集合的一个拷贝。在这之后,我们修改原有集合中的每个Employee对象的designation字段,希望这个改变不会影响到拷贝集合,但事与愿违,解决这个问题的方法是深度拷贝集合类中的元素。
- - Original Collection [Joe: Manager, Frank: Developer, Tim: Developer]
- - Copy of Collection [Joe: Manager, Frank: Developer, Tim: Developer]
- - Original Collection after modification [Joe: staff, Frank: staff, Tim: staff]
- - Copy of Collection without modification [Joe: staff, Frank: staff, Tim: staff]
可以很清楚地看到修改原有集合的Employee对象(即修改designation字段为"staff")也会反映到拷贝集合中,因为克隆是浅拷贝,它指向的是堆中同一个Employee对象。为了修复这个问题,我们需要遍历集合深度拷贝所有Employee对象,而在这之前,我们需要为Employee对象覆写clone方法。
1)让Employee类实现Cloneable接口;
2)在Employee类中添加下面的clone()方法;
- @Override
- protected Employee clone() {
- Employee clone = null;
- try{
- clone = (Employee) super.clone();
-
- }catch(CloneNotSupportedException e){
- throw new RuntimeException(e);
- }
-
- return clone;
- }
3)使用下面的代码,用Java中深度拷贝取代拷贝构造函数;
- Collection<Employee> copy = new HashSet<Employee>(org.size());
- Iterator<Employee> iterator = org.iterator();
- while(iterator.hasNext()){
- copy.add(iterator.next().clone());
- }
4)为修改后的集合运行同样的代码,会得到不同的输出:
- - Original Collection after modification [Joe: staff, Tim: staff, Frank: staff]
- - Copy of Collection without modification [Frank: Developer, Joe: Manager, Tim: Developer]
可以看出克隆后的和原有集合相互独立,分别指向不同的对象。
这便是Java如何克隆集合的所有内容,现在我们明白集合类中各种拷贝构造函数,如List或Set中addAll()方法,仅仅实现了集合的浅拷贝,即原有集合和拷贝的集合指向相同的对象。这也便要求在集合中存储任何对象,必须支持深度拷贝操作。
PS: 原Blog中的代码执行可能会遇到一些问题,但关键是这中间的思想很重要。