当前位置: 首页 > news >正文

Java SE Cloneable接口和深/浅拷贝

Java为我们提供了各种各样功能的接口,Clonable接口就是其中之一。

它通常配合Object类的 clone方法使用。这个方法可以为我们创建一个对象的拷贝,即复制一个对象。在进入本文的主要内容之前,先来对访问限定符 protected进行一个解剖。

1.再谈 protected

我们知道,在一个类中被 protected修饰的字段和方法,它们的访问权限如下:

  1. 本类内部:本类内部的所有成员都能够访问被protected修饰的元素。
  2. 同一个包内:同一包中的其他类,不管是子类还是非子类,都可以访问该元素。
  3. 不同包的子类:若子类与父类不在同一个包中,子类可以通过继承或者创建子类对象的方式来访问父类的 protected成员。

这里重点讨论第三点,当一个类中有被 protected修饰的字段和方法,并且它的子类与它在不同包时,那么子类只能在自身内部去调用父类的 protected成员,不能在外部调用。

举个例子如下:

//父类 Animal位于Demo1包
package Demo1;public class Animal {private String name;private int age;public Animal(String name, int age) {this.name = name;this.age = age;}protected void eat() {System.out.println(this.name + "正在吃饭....");}
}//子类 Dog位于Demo2包package Demo2;import Demo1.Animal;public class Dog extends Animal {public Dog(String name, int age) {super(name, age);}public void eat1() {super.eat();}
}//测试类 Testimport Demo2.Dog;public class Test {public static void main(String[] args) {Dog dog = new Dog("大黄",1);//dog.eat();发生错误:java: eat() 在 Demo1.Animal 中是 protected 访问控制dog.eat1();//正常运行,运行结果:大黄正在吃饭....}
}

我们可以发现,确实如此,eat()方法是父类Animal中被protected修饰的方法,不能在子类Dog类外部直接调用,像这样:dog.eat(),只能在子类内部调用,只有在子类内部的方法中去调用父类Animal,像这样:eat1()中的super.eat();这样Dog类外部使用 dog.eat1(),就可以间接调用父类Animal的eat()方法。

当然,我们也可以对在子类中对父类的protected方法进行重写,这样在子类外部就能直接调用了。这里就不再举例了。

 2. 如何进行克隆

要使用Object类的clone方法,我们通常需要做以下几件事:

  1. 在要克隆的类中重写Object类的 clone方法
  2. 实现 Cloneable接口
  3. 事先处理异常情况

现在有一个Person类,我们打算克隆它的一个对象。

1.在要克隆的类中重写Object类的 clone方法

package Demo3;public class Person {public String name;public Person(String name) {this.name = name;}//重写 clone方法@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}

注:可以通过快捷键“Alt + Insert”快速重写 clone方法。 

 2.实现 Cloneable接口

package Demo3;public class Person implements Cloneable{private String name;public Person(String name) {this.name = name;}@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}

注:当我们进入 Cloneable接口时,会发现它是一个空接口,它的作用是 宣布实现了这个接口的类可以克隆

 3.处理异常情况,我们创建一个测试类,在测试类中去克隆一个Person对象。

package Demo3;public class Test {public static void main(String[] args) throws CloneNotSupportedException{Person person1 = new Person("小明");Person person2 = (Person) person1.clone();}
}
//这里,需要做两件事
//1.把重写的 clone方法中的:throws CloneNotSupportedException 这个部分复制到 mian方法后面。
//2.因为重写的 clone方法的返回值为 Object,因此这里需要强转。

让我们测试一下:

package Demo3;public class Test {public static void main(String[] args) throws CloneNotSupportedException{Person person1 = new Person("小明");Person person2 = (Person) person1.clone();System.out.println("person1的名字:" + person1.name);System.out.println("person2的名字:" + person2.name);}
}//运行结果
person1的名字:小明
person2的名字:小明

显然,确实做到克隆一份person1。

那么它的过程是怎么样的呢?我们通过图画的方式来说明:

 

 3. 浅拷贝

知道了如何进行克隆,接着我们来讨论一下什么是浅拷贝,顾名思义可以理解为:浅度的克隆。

举个例子:我们在创建一个新的类 Money,在 Person类中创建一个 money对象,并将它作为一个字段。

//Money类package Demo3;public class Money {public int m = 10;
}//Person类package Demo3;public class Person implements Cloneable{public String name;public Money money = new Money();public Person(String name) {this.name = name;}@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}

现在,我们再去克隆,接着修改person2对象当中的money对象的m,看看结果如何。

package Demo3;public class Test {public static void main(String[] args) throws CloneNotSupportedException{Person person1 = new Person("小明");Person person2 = (Person) person1.clone();System.out.println("person1的m:" + person1.money.m);System.out.println("修改前的person2的m:" + person2.money.m);person2.money.m = 100;System.out.println("==================");System.out.println("person1的m:" + person1.money.m);System.out.println("修改后的person2的m:" + person2.money.m);}
}//运行结果
person1的m:10
修改前的person2的m:10
==================
person1的m:100
修改后的person2的m:100

我们可以发现,person1与person2公用一个 money.m,因为person1和person2的money引用指向同一个对象。对于它的过程,我们依旧可以用画图的方式表示:

4. 深拷贝 

 在浅拷贝的例子里,我们会发现 person1和 person2公用一个 money.m,原因是它们的 money引用中储存的是同一个money对象的地址。现在我们不想它们公用一个 money.m,那么我们该怎么做呢?这里我们就需要实现深拷贝。

想要实现深拷贝,要做两件事:

  1. 要进行拷贝的类都要重写 clone方法和实现 Cloneable接口,例如作为 Person类字段的 Money类也要重写clone方法。
  2. 修改主要的类的重写的clone方法,例如 上述例子中的 Person类,它是主要的类,因为Money类是它的字段。

 在Money类中重写 clone方法和实现 Cloneable接口。


package Demo3;public class Money implements Cloneable{public int m = 10;//重写 clone方法@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}

修改 Person类中的重写的 clone方法。

package Demo3;public class Person implements Cloneable{public String name;public Money money = new Money();public Person(String name) {this.name = name;}@Overrideprotected Object clone() throws CloneNotSupportedException {Person temp = (Person) super.clone();   temp.money = (Money) this.money.clone();return temp;}
}//说明:这里先创建一个 Person类型的变量 temp,用来接收克隆的person对象,
//接着让temp.money去接收克隆的money对象,最后返回 temp变量。

现在,我们再去克隆,接着修改person2对象当中的money对象的m,看看结果如何。

package Demo3;public class Test {public static void main(String[] args) throws CloneNotSupportedException{Person person1 = new Person("小明");Person person2 = (Person) person1.clone();System.out.println("person1的m:" + person1.money.m);System.out.println("修改前的person2的m:" + person2.money.m);person2.money.m = 100;System.out.println("==================");System.out.println("person1的m:" + person1.money.m);System.out.println("修改后的person2的m:" + person2.money.m);}
}//运行结果
person1的m:10
修改前的person2的m:10
==================
person1的m:10
修改后的person2的m:100

显然,此时修改 person2的 money.m也不会影响person1,因为person2的 money对象是克隆person1的,不再是与person1公用了。对于它的过程,我们用图画表示如下:

到此,本文完。本文若有不对的地方,还请各位看官指出,多谢!!!

相关文章:

  • 聊一聊 C# NativeAOT 多平台下的函数导出
  • day10机器学习的全流程
  • Python入门手册:模块和包的导入与使用
  • 基于SpringBoot开发一个MCP Server
  • 社区造数服务接入MCP|得物技术
  • JavaScript 中 this 指向全解析:从基础到 Vue 应用
  • C语言 文件操作(2)
  • Nodejs+http-server 使用 http-server 快速搭建本地图片访问服务
  • 不同坐标系下的 面积微元
  • 越南跨境电商免税政策遇冷?工商会为何踩下“刹车”
  • 8086 处理器寄存器超详细解析:从原理到实战
  • BEV和OCC学习-1:数据集以及评估指标
  • 麒麟v10,arm64架构,编译安装Qt5.12.8
  • Python爬虫实战:研究Tornado框架相关技术
  • UDP组播套接字与URI/URL/URN技术详解
  • Qt/C++开发监控GB28181系统/sip协议/同时支持udp和tcp模式/底层协议解析
  • 一文了解智慧教育顶刊《Journal of Computer Assisted Learning》2025年4月研究热点
  • 上传头像upload的简易方法,转base64调接口的
  • RK3568DAYU开发板-平台驱动开发:I2C驱动(原理、源码、案例分析)
  • BeeWorks:构建企业智能数字化协作中枢,实现办公与业务系统的无缝整合
  • 企业网站内使用了哪些网络营销方式/乔拓云网站注册
  • app开发流程详解/惠州seo
  • 常规网站建设价格实惠/搜狗指数
  • 网站建设昆明/百度风云榜排行榜
  • 网站建设 昆明 价格/百度指数怎么算
  • 营销型网站建设哪家便宜/seo优化工作