第一种:http://blog.chinaunix.net/uid-20577907-id-3132181.html   (亲测有用)

java 深度复制 2012-03-13 20:25:21

分类: Java

最近的系统改造,遇到了前人写的PropertyUtilsBean. copyProperties ( Object dest, Object orig)
方法来复制对象,想到自己对Java API了解到太少,以后可以多多学习。
由于改造后的系统运用JPA技术,将以前的JOPO转换成了实体Bean,并且实体Bean间建立起了表之间的关联关系(一对多、多对一、多对多之乎者也的),调试程序时发现copyProperties方法失效了,不解之下,找到了JavaSE的API一窥究竟,原来该方法不支持续对象的嵌套复制,API中如是说:
Note that this method is intended to perform a "shallow copy" of the properties and so complex properties (for example, nested ones) will not be copied.

Note, that this method will not copy a List to a List, or an Object[] to an Object[]. It's specifically for copying JavaBean properties.

于是在网上借鉴了用输入输出流进行深度复制的方法,问题终于解决了,在此感谢豁然开朗的代码:

点击(此处)折叠或打开

  1. private static Object depthClone(Object srcObj){
  2. 74. Object cloneObj = null;
  3. 75. try {
  4. 76. ByteArrayOutputStream out = new ByteArrayOutputStream();
  5. 77. ObjectOutputStream oo = new ObjectOutputStream(out);
  6. 78. oo.writeObject(srcObj);
  7. 79.
  8. 80. ByteArrayInputStream in = new ByteArrayInputStream(out.toByteArray());
  9. 81. ObjectInputStream oi = new ObjectInputStream(in);
  10. 82. cloneObj = oi.readObject();
  11. 83. } catch (IOException e) {
  12. 84. e.printStackTrace();
  13. 85. } catch (ClassNotFoundException e) {
  14. 86. e.printStackTrace();
  15. 87. }
  16. 88. return cloneObj;
  17. 89. }

第二种:http://www.importnew.com/10761.html

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

2014/04/14 | 分类: 基础技术 | 2 条评论

分享到: 6
本文由 ImportNew - hejiani 翻译自 javarevisited。欢迎加入 翻译小组。转载请见文末要求。

编程人员经常误用各个集合类提供的拷贝构造函数作为克隆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
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
import java.util.Collection;
import java.util.HashSet;
import java.util.Iterator;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Java program to demonstrate copy constructor of Collection provides shallow
* copy and techniques to deep clone Collection by iterating over them.
* @author http://javarevisited.blogspot.com
*/
public class CollectionCloningTest {
     private static final Logger logger = LoggerFactory.getLogger(CollectionCloningclass);
     public static void main(String args[]) {
         // deep cloning Collection in Java
         Collection<Employee> org = new HashSet<>();
         org.add( new Employee( "Joe" , "Manager" ));
         org.add( new Employee( "Tim" , "Developer" ));
         org.add( new Employee( "Frank" , "Developer" ));
         // creating copy of Collection using copy constructor
         Collection<Employee> copy = new HashSet<>(org);
         logger.debug( "Original Collection {}" , org);
         logger.debug( "Copy of Collection {}" , copy );
         
         Iterator<Employee> itr = org.iterator();
         while (itr.hasNext()){
             itr.next().setDesignation( "staff" );
         }
         logger.debug( "Original Collection after modification {}" , org);
         logger.debug( "Copy of Collection without modification {}" , copy );
         // deep Cloning List in Java
     }
}
class Employee {
     private String name;
     private String designation;
     
     public Employee(String name, String designation) {
         this .name = name;
         this .designation = designation;
     }
     
     public String getDesignation() {
         return designation;
     }
     public void setDesignation(String designation) {
         this .designation = designation;
     }
     public String getName() {
         return name;
     }
     public void setName(String name) {
         this .name = name;
     }
     @Override
     public String toString() {
         return String.format( "%s: %s" , name, designation );
     }
}

输出:

1
2
3
4
- 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]

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

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

1
2
3
4
5
6
7
8
9
10
11
12
@Override
     protected Employee clone() {
         Employee clone = null ;
         try {
             clone = (Employee) super .clone();
         } catch (CloneNotSupportedException e){
             throw new RuntimeException(e); // won't happen
         }
          
         return clone;
     }

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

1
2
3
4
5
6
Collection<Employee> copy = new HashSet<Employee>(org.size());
Iterator<Employee> iterator = org.iterator();
while (iterator.hasNext()){
     copy.add(iterator.next().clone());
}

4)运行相同的代码更改原始集合,克隆集合不会也被更改。

1
2
- Original Collection after modification  [Joe: staff, Tim: staff, Frank: staff]
- Copy of Collection without modification [Frank: Developer, Joe: Manager, Tim: Developer]

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

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

java深度复制 xjh 亲测 两种方法相关推荐

  1. Java中的string定义的两种方法和区别

    java中的String定义的两种方法和区别 第一种:new方式 String s1 = new String("hello world"); String s2 = new St ...

  2. Java生成二维码的两种方法

    本文介绍Java生成二维码的两种方法,这两种方法都依赖于google提供的二维码依赖包. 一种是自己写工具类,代码可以网上抄. 另一种是使用hutool第三方工具类的依赖包,不用自己抄代码. 一.自定 ...

  3. Java 跨平台获取 MAC 地址的两种方法

    前言 有时我们因为软件授权或者其它需要获取主机唯一标识而需要获取用户主机的MAC地址,而本文则将介绍如何通过Java来实现跨平台获取MAC地址的两种方法,同时具体的代码也已上传到GitHub. 如果不 ...

  4. java对文件进行压缩的两种方法

    在工作中,我们或多或少都会接触到文件的压缩和解压,在window系统中,我们只需下载一个能对文件进行解压缩的应用即可,但如果让我们自己动手写对文件压缩的代码,顿时就头大了. 在java中,我们都知道输 ...

  5. Java实现条件性参加会议问题(两种方法)

    有人邀请A,B,C,D,E,F 6个人参加一项会议,这6个人有些奇怪,因为他们有很多要求,已知: 1)A,B两人至少有1人参加会议:   2)A,E,F 3人中有2人参加会议:   3)B和C两人一致 ...

  6. java中调用dll文件的两种方法

    https://www.cnblogs.com/huozhong/p/5830485.html JNA地址:http://blog.csdn.net/shendl/article/details/35 ...

  7. Java数组实现循环队列的两种方法

    用java实现循环队列的方法: 1.增加一个属性size用来记录目前的元素个数.目的是当head=rear的时候,通过size=0还是size=数组长度,来区分队列为空,或者队列已满. 2.数组中只存 ...

  8. Java的JFrame窗体的创建(两种方法)

    第一种直接创建 package com.jwz.h综合项目;import javax.swing.*;public class Test {public static void main(String ...

  9. 在DWR中实现直接获取一个JAVA类的返回值的两种方法

    第一种实现(来源网上转贴): js 代码 function Test() { var _data = ""; this.getString = function() { //设置成 ...

最新文章

  1. python中系列的含义_一篇文章让你彻底搞清楚Python中self的含义
  2. 我校四名学生在全国中学生物理竞赛中勇夺一金三银并全部直保清华
  3. linux安装软件的几种方式(kali平台)和一些实用的软件(持续更新)
  4. 折叠屏来了,开发者们准备好了吗?
  5. SAP UI5 setProperty 的执行逻辑单步调式和分析
  6. 【HTML】JS基础知识
  7. 用ajax替换html代码,替换Ajax响应一个div的内部HTML(Replace inner HTML of a div w
  8. 用calloc()函数分配内存
  9. XMLHttpRequestEventTarget
  10. pygame系列文章
  11. c#简要概括面向对象的三大特征(三)
  12. MATLAB实现imrotate函数
  13. 力扣438. 找到字符串中所有字母异位词 C++ (滑动窗口 + 数组)
  14. Hexo + yilia 主题实现文章目录
  15. ECCV2022 Workshop | 复杂环境中的多目标跟踪和分割
  16. html消防产品模版,消防设备项目建议书模板
  17. echarts:从github及其镜像下载china.js和china.json
  18. iview tooltip: true, 处理文字,溢出用点代替
  19. 【办公类-16-06】“2022下学期 总园活动室(沙水+建构)排班表”(python 排班表系列)
  20. 介紹HyperSnap使用方法!

热门文章

  1. AutoCAD绘制表格的插件
  2. aapt 命令生成 R文件
  3. 联邦学习顶刊论文整理,如何高效学论文?
  4. 4. TabBar,ScrollBar,ProgressBar,MenuBar,ToolBar
  5. ZLMediaKit + wvp-GB28181-pro部署(windows64位)
  6. luogu 2344
  7. 惊雷用计算机唱歌,除了《惊雷》还有多少喊麦神曲?这十首神作你一定听过!...
  8. 十六进制字符串转十进制字符串
  9. 通过设置让火狐浏览器自动清缓存,不需要自己每次手动清除了
  10. MFC EnableWindow() 启用和禁用控件(设置个控件或窗口可用/不可用),ShowWindow()使某个控件或窗口(显示/不显示)可见/不可见