Java面向对象3-深拷贝与浅拷贝
深拷贝和浅拷贝最根本的区别在于是否真正获取一个对象的复制实体,而不是引用。
假设B复制了A,修改A的时候,看B是否发生变化:
如果B跟着也变了,说明是浅拷贝,拿人手短!(修改堆内存中的同一个值)
如果B没有改变,说明是深拷贝,自食其力!(修改堆内存中的不同的值)
1、浅拷贝
在拷贝一个对象时,对对象的基本数据类型的成员变量进行拷贝,但对引用类型的成员变量只进行引用的传递,并没有创建一个新的对象,当对引用类型的内容修改会影响被拷贝的对象。
只是增加了一个指针指向已存在的内存地址,java中clone方法是一个浅拷贝,引用类型依然在传递引用。
如果克隆对象的子对象是不可变的,或者子对象没有更改器方法,那么就是安全的
2、深拷贝
除了对基本数据类型的成员变量进行拷贝,对引用类型的成员变量进行拷贝时,创建一个新的对象来保存引用类型的成员变量。
增加了一个指针并且申请了一个新的内存,使这个增加的指针指向这个新的内存。
重新定义clone方法,克隆出所有子对象
Cloneable接口出现和接口的使用没有任何关系,因为clone方法是Object类继承而来的
Cloneable接口是标记接口,不含任何方法,唯一的作用就是允许在类型查询中视同instanceof
实现深拷贝有两种方法:
(1)序列化该对象,然后反序列化回来,就能得到一个新的对象了。
序列化:将对象写入到IO流中; 反序列化:从IO流中恢复对象 序列化机制允许将实现序列化的java对象转化为字节序列,这些字节序列可以保存到磁盘或者网络传输上,以达到以后恢复成原来的对象,序列化机制使得对象可以脱离程序的运行而独立存在。
(2)继续利用clone()方法,对该对象的引用类型变量再实现一次clone()方法。
(1)序列化
1 public class Student3 implements Serializable,Cloneable{ 2 private static final long serialVersionUID = 3462139480068147262L; 3 private Integer age; 4 private String name; 5 6 public Student3(Integer age, String name) { 7 this.age = age; 8 this.name = name; 9 } 10 11 public Integer getAge() { 12 return age; 13 } 14 15 public void setAge(Integer age) { 16 this.age = age; 17 } 18 19 public String getName() { 20 return name; 21 } 22 23 public void setName(String name) { 24 this.name = name; 25 } 26 27 @Override 28 protected Object clone() throws CloneNotSupportedException { 29 return super.clone(); 30 } 31 32 public static void main(String[] args) throws CloneNotSupportedException { 33 File file = new File("D:/test.txt"); 34 Student3 stu = new Student3(18, "xiaoxian"); 35 36 System.out.println("clone方法是浅拷贝"); 37 Student3 clone = (Student3)stu.clone(); 38 System.out.println("clone == stu的结果:"+ (clone==stu)); 39 System.out.println("clone.name == stu.name的结果:"+ (clone.name==stu.name)); 40 41 System.out.println("将对象序列化是深拷贝"); 42 //将对象序列化到IO流中 43 try { 44 ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream(file)); 45 objectOutputStream.writeObject(stu); 46 objectOutputStream.close(); 47 } catch (IOException e) { 48 e.printStackTrace(); 49 } 50 51 //将对象从IO流中反序列化出来 52 try { 53 ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(file)); 54 Student3 student3 = (Student3) objectInputStream.readObject(); 55 System.out.println("student3 == stu的结果:"+(stu == student3)); 56 System.out.println("student3.name == stu.name的结果:"+(stu.name == student3.name)); 57 } catch (Exception e) { 58 e.printStackTrace(); 59 } 60 } 61 62 }
(2)重写clone方法
1 package com.company.DeepCopy; 2 3 public class DeepCopy { 4 5 6 public static void main(String[] args) { 7 Age a = new Age(20); 8 Student stu1 = new Student("宁采臣",a,174); 9 10 //通过调用重写后的clone方法进行浅拷贝 11 Student stu2 = (Student)stu1.clone(); 12 System.out.println(stu1.toString()); 13 System.out.println(stu2.toString()); 14 System.out.println(); 15 16 //尝试修改stu1中的各属性,观察stu2的属性有没有变化 17 stu1.setName("聂小倩"); 18 //改变age这个引用类型的成员变量的值 19 a.setAge(18); 20 //stu1.setaAge(new Age(99)); 使用这种方式修改age属性值的话,stu2是不会跟着改变的。因为创建了一个新的Age类对象而不是改变原对象的实例值 21 stu1.setLength(157); 22 System.out.println(stu1.toString()); 23 System.out.println(stu2.toString()); 24 } 25 }
1 package com.company.DeepCopy; 2 3 public class Student implements Cloneable { 4 //学生类的成员变量(属性),其中一个属性为类的对象 5 private String name; 6 private Age aage; 7 private int length; 8 //构造方法,其中一个参数为另一个参数的对象 9 public Student(String name,Age a,int length){ 10 this.name = name; 11 this.aage = a; 12 this.length = length; 13 } 14 public String getName() { 15 return name; 16 } 17 public void setName(String name) { 18 this.name = name; 19 } 20 public Age getAge() { 21 return this.aage; 22 } 23 public void setAge(Age age) { 24 this.aage = age; 25 } 26 public int getLength() { 27 return this.length; 28 } 29 public void setLength(int length) { 30 this.length = length; 31 } 32 public String toString(){ 33 return "姓名是:"+this.getName()+",年龄为:"+this.getAge().toString()+",长度是:"+this.getLength(); 34 } 35 //重写Object类的clone方法 36 public Object clone(){ 37 Object obj = null; 38 //调用Object类的clone方法---浅拷贝 39 try { 40 obj = super.clone(); 41 }catch (CloneNotSupportedException e){ 42 e.printStackTrace(); 43 } 44 //调用Age类的clone方法进行深拷贝 45 //先将obj转化为学生类实例 46 Student stu = (Student)obj; 47 //学生类实例的Age对象属性,调用其clone方法进行拷贝 48 stu.aage=(Age)stu.getAge().clone(); 49 return obj; 50 } 51 }
1 package com.company.DeepCopy; 2 3 public class Age implements Cloneable{ 4 //年龄类的成员变量(属性) 5 private int age; 6 //构造方法 7 public Age(int age){ 8 this.age = age; 9 } 10 public int getAge(){ 11 return age; 12 } 13 public void setAge(int age) { 14 this.age = age; 15 } 16 @Override 17 public String toString() { 18 return this.age+""; 19 } 20 //重写Object的clone方法 21 public Object clone() { 22 Object obj = null; 23 try { 24 obj = super.clone(); 25 } catch (CloneNotSupportedException e) { 26 e.printStackTrace(); 27 } 28 return obj; 29 } 30 }
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· DeepSeek “源神”启动!「GitHub 热点速览」
· 微软正式发布.NET 10 Preview 1:开启下一代开发框架新篇章
· 我与微信审核的“相爱相杀”看个人小程序副业
· C# 集成 DeepSeek 模型实现 AI 私有化(本地部署与 API 调用教程)
· spring官宣接入deepseek,真的太香了~