当前位置:   article > 正文

java clone 中的浅复制和深复制

java clone 中的浅复制和深复制

什么是浅复制?

首先,浅复制会创建一个新对象,这个新的对象各个字段的值会从原始对象复制过来,如果某个字段是引用其他的对象,那么仅仅复制此对象在内存中的引用地址。

图1

如图1 所示,对象 MainObject1 有一个 int 类型的字段 Field1 和一个字段引用 ContainObject1对象,被克隆的新对象是 MainObject2, MainObject2 有一个 int 类型字段 Field2, Field2 的值是从对象MainObject1的字段 Field1 复制过来,而MainObject2的一个字段和MainObject1 中的某个字段仍然指向同一个对象ContainObject1,也就是说只要ContainObject1 发生任何变化,MainObject1 和MainObject2所引用的ContainObject1 都会用变化。

什么是深复制?

深复制也会创建一个新的对象,除了复制这个新对象里的原始类型字段的值,还要对此对象的引用字段再做克隆,而不是仅仅复制此引用字段再内存中的引用地址。

图2

如图2 所示,MainObject1 是原始对象,MainObject2 是被克隆出的对象,从图2和图1的差别可以看出,克隆对象MainObject2 字段ContainObject2和 MainObject1 里的字段 ContainObject1 指向的是不同的对象,也就是说,MainObject1 里的 ContainObject1 发生任何变化,都不会影响到 MainObject2 里的 ContainObject2。

用 java 实现浅复制示例:

  1. class Subject {
  2. private String name;
  3. public String getName() {
  4. return name;
  5. }
  6. public void setName(String s) {
  7. name = s;
  8. }
  9. public Subject(String s) {
  10. name = s;
  11. }
  12. }
  13. class Student implements Cloneable {
  14. //Contained object
  15. private Subject subj;
  16. private String name;
  17. public Subject getSubj() {
  18. return subj;
  19. }
  20. public String getName() {
  21. return name;
  22. }
  23. public void setName(String s) {
  24. name = s;
  25. }
  26. public Student(String s, String sub) {
  27. name = s;
  28. subj = new Subject(sub);
  29. }
  30. public Object clone() {
  31. //shallow copy
  32. try {
  33. return super.clone();
  34. } catch (CloneNotSupportedException e) {
  35. return null;
  36. }
  37. }
  38. }
  39. public class CopyTest {
  40. public static void main(String[] args) {
  41. //Original Object
  42. Student stud = new Student("John", "Algebra");
  43. System.out.println("Original Object: " + stud.getName() + " - "
  44. + stud.getSubj().getName());
  45. //Clone Object
  46. Student clonedStud = (Student) stud.clone();
  47. System.out.println("Cloned Object: " + clonedStud.getName() + " - "
  48. + clonedStud.getSubj().getName());
  49. stud.setName("Dan");
  50. stud.getSubj().setName("Physics");
  51. System.out.println("Original Object after it is updated: "
  52. + stud.getName() + " - " + stud.getSubj().getName());
  53. System.out.println("Cloned Object after updating original object: "
  54. + clonedStud.getName() + " - " + clonedStud.getSubj().getName());
  55. }
  56. }
输出结果是:

  1. Original Object: John - Algebra
  2. Cloned Object: John - Algebra
  3. Original Object after it is updated: Dan - Physics
  4. Cloned Object after updating original object: John - Physics

用java 实现深复制示例

只需要把上例的Student类的clone 方法修改如下:

  1. class Student implements Cloneable {
  2. //Contained object
  3. private Subject subj;
  4. private String name;
  5. public Subject getSubj() {
  6. return subj;
  7. }
  8. public String getName() {
  9. return name;
  10. }
  11. public void setName(String s) {
  12. name = s;
  13. }
  14. public Student(String s, String sub) {
  15. name = s;
  16. subj = new Subject(sub);
  17. }
  18. public Object clone() {
  19. //Deep copy
  20. Student s = new Student(name, subj.getName());
  21. return s;
  22. }
  23. }
修改后运行结果如下:

  1. Original Object: John - Algebra
  2. Cloned Object: John - Algebra
  3. Original Object after it is updated: Dan - Physics
  4. Cloned Object after updating original object: John - Algebra


使用序列化实现深复制

请看如下示例:

  1. public class ColoredCircle implements Serializable {
  2. private int x;
  3. private int y;
  4. public ColoredCircle(int x, int y) {
  5. this.x = x;
  6. this.y = y;
  7. }
  8. public int getX() {
  9. return x;
  10. }
  11. public void setX(int x) {
  12. this.x = x;
  13. }
  14. public int getY() {
  15. return y;
  16. }
  17. public void setY(int y) {
  18. this.y = y;
  19. }
  20. @Override
  21. public String toString() {
  22. return "x=" + x + ",y=" + y;
  23. }
  24. }

  1. public class DeepCopy {
  2. static public void main(String[] args) {
  3. ObjectOutputStream oos = null;
  4. ObjectInputStream ois = null;
  5. try {
  6. // create original serializable object
  7. ColoredCircle c1 = new ColoredCircle(100, 100);
  8. // print it
  9. System.out.println("Original = " + c1);
  10. ColoredCircle c2 = null;
  11. // deep copy
  12. ByteArrayOutputStream bos = new ByteArrayOutputStream();
  13. oos = new ObjectOutputStream(bos);
  14. // serialize and pass the object
  15. oos.writeObject(c1);
  16. oos.flush();
  17. ByteArrayInputStream bin =
  18. new ByteArrayInputStream(bos.toByteArray());
  19. ois = new ObjectInputStream(bin);
  20. // return the new object
  21. c2 = (ColoredCircle) ois.readObject();
  22. // verify it is the same
  23. System.out.println("Copied = " + c2);
  24. // change the original object's contents
  25. c1.setX(200);
  26. c1.setY(200);
  27. // see what is in each one now
  28. System.out.println("Original = " + c1);
  29. System.out.println("Copied = " + c2);
  30. } catch (Exception e) {
  31. System.out.println("Exception in main = " + e);
  32. } finally {
  33. try {
  34. oos.close();
  35. ois.close();
  36. } catch (IOException e) {
  37. System.out.println(e);
  38. }
  39. }
  40. }
  41. }

程序输出结果:

  1. Original = x=100,y=100
  2. Copied = x=100,y=100
  3. Original = x=200,y=200
  4. Copied = x=100,y=100



声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/从前慢现在也慢/article/detail/616639
推荐阅读
相关标签
  

闽ICP备14008679号