在日常的开发中,我们经常会遇到需要克隆一个对象,对克隆的对象操作之后不影响原有对象以及与之关联的对象

Java的Object类提供了clone方法,用来进行克隆对象,不过JDK提供的clone()方法只是浅度克隆,要想深度克隆需要自己来实现,那么什么是浅度克隆和深度克隆呢?

浅度克隆

JDK提供的克隆是浅度克隆,它只将对象中的基础数据类型的成员变量克隆到新对象中,对象中的引用类型只是克隆了一个引用,克隆后的引用类型还是指向原对象,如下图所示

代码如下:

public class Teacher {private String name;private int age;public Teacher(String name,int age){this.name = name;this.age = age;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}
}
public class User implements Cloneable{private String name;private int age;private Teacher teacher;public User(String name,int age){this.name = name;this.age = age;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public Teacher getTeacher() {return teacher;}public void setTeacher(Teacher teacher) {this.teacher = teacher;}@Overrideprotected User clone() throws CloneNotSupportedException {User user = null;try {user = (User) super.clone();} catch (CloneNotSupportedException e){e.printStackTrace();}return user;}
}
public class TestClone {public static void main(String[] args) throws CloneNotSupportedException {User user = new User("Tom",13);user.setTeacher(new Teacher("Jack",40));User userClone = user.clone();userClone.getTeacher().setName("Jerry");userClone.setName("Tom1");userClone.setAge(20);System.out.println(user.getName() + "|" + user.getAge() + "|" + user.getTeacher().getName());System.out.println(userClone.getName() + "|" + userClone.getAge() + "|" + userClone.getTeacher().getName());}
}

输出结果:

Tom|13|Jerry
Tom1|20|Jerry

从输出结果可以看出,克隆对象修改name和age不会影响原对象,但对引用类型Teacher做修改,就会影响原对象

深度克隆

深度克隆就是把整个对象完全克隆一份,包括对象中的引用类型,如图所示

JDK没有提供尝试克隆的方法,只能我们自己来实现,把User类的clone()方法自己实现一下就可以实现简单的尝试克隆了

 @Overrideprotected User clone() throws CloneNotSupportedException {User user = null;try {user = (User) super.clone();//克隆的时候new一个Teacher对象,就可以实现深度克隆Teacher te = new Teacher(this.teacher.getName(),this.teacher.getAge());user.setTeacher(te);} catch (CloneNotSupportedException e){e.printStackTrace();}return user;}

这样就可以实现User对象的深度克隆了,是不是很简单?

你以为这样就学会了深度克隆,那就真的太天真了,这里我只列举了一个最简单的对象

实际在开发过程中遇到的对象远远比这个要复杂的多,可能包含多层对象的嵌套,如果都自己去实现的话,那就太麻烦了,而且容易出现BUG

一种比较常用的实现方式是利用Java的序列化和反序列化

对象序列化必须实现Serializable接口

public class Teacher implements Serializable {private String name;private int age;public Teacher(String name,int age){this.name = name;this.age = age;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}
}
public class Teacher implements Serializable {private String name;private int age;public Teacher(String name,int age){this.name = name;this.age = age;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}
}
public class TestClone {public static void main(String[] args) throws IOException, ClassNotFoundException {User user = new User("Tom",13);user.setTeacher(new Teacher("Jack",40));//序列化对象ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();ObjectOutputStream objectOutputStream = new ObjectOutputStream(byteArrayOutputStream);objectOutputStream.writeObject(user);//反序列化对象ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(byteArrayOutputStream.toByteArray());ObjectInputStream objectInputStream = new ObjectInputStream(byteArrayInputStream);User cloneUser = (User) objectInputStream.readObject();cloneUser.setName("Tom_clone");cloneUser.setAge(20);cloneUser.getTeacher().setName("Jack_clone");System.out.println(user.getName() + "|" + user.getAge() + "|" + user.getTeacher().getName());System.out.println(cloneUser.getName() + "|" + cloneUser.getAge() + "|" + cloneUser.getTeacher().getName());}
}

输出结果:

Tom|13|Jack
Tom_clone|20|Jack_clone

可以看到已经完成了深度复杂,对引用类型的修改不会影响到原对象,这种方式的好处是不管对象嵌套多少层,对象进行序列化和反序列化之后,都会产生一个全新的对象

注意:static 和 transient 类型的变量不会被序列化,所以这两种类型的深度克隆不能使用序列化的方式

如果感觉对你有些帮忙,请收藏好,你的关注和点赞是对我最大的鼓励!
如果想跟我一起学习,坚信技术改变世界,请关注【Java天堂】公众号,我会定期分享自己的学习成果,第一时间推送给您

啥是对象的浅度克隆和深度克隆?相关推荐

  1. jquery,angular 对象数组的克隆和深度克隆

    var obj1 = $.extend(false, {}, obj2);  //对象克隆 var obj1 = $.extend(true, {}, obj2);   //对象深度克隆 ------ ...

  2. Java的“影子克隆”和“深度克隆”

    今天来学习学习java对象的克隆,在写代码的时候,有时候我们会这样写:对象1=对象2,也就是把对象2赋值给对象1了,但是这样做有个问题,就是如果我们修改了对象2的属性值,对象1的相同属性值也被修改了, ...

  3. Cloneable接口的作用与深入理解深度克隆与浅度克隆

    cloneable接口的作用 cloneable其实就是一个标记接口,只有实现这个接口后,然后在类中重写Object中的clone方法,然后通过类调用clone方法才能克隆成功,如果不实现这个接口,则 ...

  4. Java深度克隆对象

    关于对象的克隆,Java自带了一个clone()方法. 但是众所周知,这个clone()方法只是做浅层的对象复制,仅适用于对象属性是简单数据类型(int/long/float等)的场景.对于对象属性是 ...

  5. javascript 数组和对象的浅复制和深度复制 assign/slice/concat/JSON.parse(JSON.stringify())...

    javascript 数组和对象的浅度复制和深度复制 在平常我们用 '='来用一个变量引用一个数组或对象,这里是'引用'而不是复制下面我们看一个例子引用和复制是什么概念 var arr=[1,2,3, ...

  6. java浅度克隆_java对象 深度克隆(不实现Cloneable接口)和浅度克隆

    为什么需要克隆: 在实际编程过程中,我们常常要遇到这种情况:有一个对象A,在某一时刻A中已经包含了一些有效值,此时可能会需要一个和A完全相同新对象B,并且此后对B任何改动都不会影响到A中的值,也就是说 ...

  7. javascript 深度克隆对象

    js一般有两种不同数据类型的值: 基本类型(包括undefined,Null,boolean,String,Number),按值传递: 引用类型(包括数组,对象),按址传递,引用类型在值传递的时候是内 ...

  8. java递归深度克隆_递归方法实现深度克隆原理:遍历对象、数组直到里边都是基本数据类型,然后再去复制,就是深度拷贝。...

    手写递归方法 //定义检测数据类型的功能函数 function checkedType(target) { return Object.prototype.toString.call(target). ...

  9. [js] 请写一个性能最好的深度克隆对象的方法

    [js] 请写一个性能最好的深度克隆对象的方法 const deepClone = (obj) => {const copy = obj instance Array ? [] : {};for ...

最新文章

  1. 数据分析模块pandas
  2. welcome-file-list标签的控制作用以及在springmvc中此标签的的配置方式
  3. 位运算应用及其注意事项
  4. mac 查看端口的使用情况
  5. 论发SCI论文和生孩子的共同点:那我这篇怀的也太久了!
  6. 火狐浏览器Json插件(JSONView)
  7. Serializer的功能
  8. 3D立体显示大屏幕拼接视频墙系统解决方案
  9. sqlite3 error: database is locked
  10. C++打印机首选项设置
  11. OOP,Object Oriented Programming 面向对象的三大特性 五大基本原则
  12. java搜索页面历史记录,页面缓存的操作(搜索历史记录),页面搜索功能实现...
  13. 上海市住房公积金销户提取的步骤
  14. Android绘图(二)使用 Graphics2D 实现动态效果
  15. win10注册mscomm32.ocx失败
  16. 域用户不准更改计算机名,请教高手,域环境下如何不退域修改计算机名?
  17. 英语口语练习系列-C41-食物词汇-鹊桥仙
  18. 代码详解设计模式--观察者模式
  19. 关于 Java 的线程状态
  20. b500k电位器引脚接法_电位器的种类、作用及其引脚的接法

热门文章

  1. 判断sqlite数据库中表是否存在的方法
  2. 镖狮网裴向宇谈互联网营销的创业之路
  3. 全站仪如何操作点到直线的测量?
  4. camera isp(Image Signal Processor)
  5. 小学计算机房安全管理制度,小学微机室安全管理制度
  6. 怎样快速掌握WORD的常用功能,看这里就够了,常见的一些word使用小技巧
  7. 使用 mulan-1.5.0 如何构造.arff文件
  8. 找茬小游戏,uniapp小程序端+pc端操作静态页
  9. 智络会员管理软件的主要特点
  10. Docx4j生成word表格(无问题),用Libreoffice转换为pdf格式乱掉问题