当前位置: 首页 > 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公用了。对于它的过程,我们用图画表示如下:

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

http://www.xdnf.cn/news/687385.html

相关文章:

  • 水域应急救援可视化平台
  • 【前端】Vue3+elementui+ts,TypeScript Promise<string>转string错误解析,习惯性请出DeepSeek来解答
  • 国产SOC有哪些?
  • 即插即用的全新算法改进策略——引导学习策略:一种用于元启发式算法设计和改进的新型更新机制
  • Unity对象池插件Lean Pool学习笔记
  • android 图片背景毛玻璃效果实现
  • Flutter 与 Android 原生布局组件对照表(完整版)
  • TencentOSTiny
  • 【模型显著性分析】配对样本 t 检验
  • 虚拟与现实时空认知同步的核心指标
  • maven中的maven-resources-plugin插件详解
  • 部署LVS-DR群集
  • Docker部署Spark大数据组件:配置log4j日志
  • Vue开发系列——零基础HTML引入 Vue.js 实现页面之间传参
  • Kotlin 中的数据类型有隐式转换吗?为什么?
  • 天津工作机会:技术文档工程师 - 华海清科股份有限公司
  • 【Linux】分页式存储管理:深刻理解页表映射
  • 【Doris基础】Apache Doris 基本架构深度解析:从存储到查询的完整技术演进
  • 金砖国家人工智能高级别论坛在巴西召开,华院计算应邀出席并发表主题演讲
  • 960g轻薄本,把科技塞进巧克力盒子
  • 从零开始学安全:服务器被入侵后的自救指南
  • 第二章 1.5 数据采集安全风险防范之数据采集安全管理
  • git和gitee的常用语句命令
  • JS语言基础
  • LiveNVR 直播流拉转:Onvif/RTSP/RTMP/FLV/HLS 支持海康宇视天地 SDK 接入-视频广场页面集成与视频播放说明
  • 实验设计与分析(第6版,Montgomery)第3章单因子实验:方差分析3.11思考题3.7 R语言解题
  • RuoYi前后端分离项目集成magic-api,并继承RuoYi的权限认证体系来管理magic-api
  • mongodb集群之副本集
  • IP证书的作用与申请全解析:从安全验证到部署实践
  • 【数据集】无缝1 km地表温度数据集(US)