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.pswp.cn/news/907263.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Python学习(3) ----- Python的函数定义及其使用

Python 中函数是组织好的、可重复使用的代码块,用于实现单一或相关联的功能。下面是函数定义和使用的完整说明: 📌 一、函数定义语法 def 函数名(参数1, 参数2默认值, *args, **kwargs):"""函数说明文档"""函…

vue2使用el-tree实现两棵树间节点的拖拽复制

原文链接&#xff1a;两棵el-tree的节点跨树拖拽实现 参照这篇文章&#xff0c;把它做成组件&#xff0c;新增左侧树&#xff08;可拖出&#xff09;被拖节点变灰提示&#xff1b; 拖拽中&#xff1a; 拖拽后&#xff1a; TreeDragComponent.vue <template><!-- …

智变与重构:AI 赋能基础教育教学的范式转型研究报告

一、研究背景与核心价值 &#xff08;一&#xff09;技术驱动下的教育转型浪潮 在全球数字化转型加速的背景下&#xff0c;人工智能作为核心技术力量&#xff0c;正重塑基础教育生态。据《人工智能赋能未来教育研究报告》指出&#xff0c;我国教育数字化战略行动已推动超 70…

Go语言中Print、Printf和Println的区别及使用场景详解

在Go语言的fmt包中&#xff0c;Print、Printf和Println是三个基础但功能各异的输出函数。本文将从多个维度进行详细对比分析&#xff0c;并给出具体的使用建议。 1. 核心区别深度解析 1.1. 函数签名与基本行为 func Print(a ...interface{}) (n int, err error) func Printf…

高端制造行业 VMware 替代案例合集:10+ 头部新能源、汽车、半导体制造商以国产虚拟化支持 MES、PLM 等核心应用系统

在“中国制造 2025”政策的推动下&#xff0c;国内的新能源、汽车制造、半导体、高端装备等高端制造产业迎来了蓬勃发展&#xff0c;成为全球制造业版图中举足轻重的力量。订单数量的激增与国产化转型的趋势&#xff0c;也为高端制造企业的 IT 基础设施带来了新的挑战&#xff…

Spring Ai | 从零带你一起走进AI项目(中英)

目录 Thinking Study question pox.xml Maven Gradle Configure API Key Use the AI Client Question Thinking 让数据变得更加贴近用户的想法 Study question null pox.xml 添加依赖 Maven <dependencies><dependency><groupId>org.springfram…

LiveGBS作为下级平台GB28181国标级联2016|2022对接海康大华宇视华为政务公安内网等GB28181国标平台查看级联状态及会话

LiveGBS作为下级平台GB28181国标级联2016|2022对接海康大华宇视华为政务公安内网等GB28181国标平台查看级联状态及会话 1、GB/T28181级联概述2、搭建GB28181国标流媒体平台3、获取上级平台接入信息3.1、向下级提供信息3.2、上级国标平台添加下级域3.3、接入LiveGBS示例 4、配置…

卸载 Office PLUS

Office PLUS作为微软官方推出的智能办公提效工具&#xff0c;自2015年问世以来&#xff0c;凭借其丰富的模板资源和便捷的智能功能&#xff0c;迅速赢得了广大职场人士和学生的青睐。本文将全面介绍Office PLUS的发展历程、核心功能、可能带来的使用问题&#xff0c;以及如何彻…

影响沉金价格的因素如何体现在多层电路板制造上?

随着科技的不断发展&#xff0c;电子产品越来越普及&#xff0c;对电路板的需求也越来越大。多层电路板作为电子产品的核心部件&#xff0c;其性能和质量直接影响到整个产品的稳定性和可靠性。在多层电路板的生产过程中&#xff0c;沉金工艺是一种常用的表面处理方法&#xff0…

扩展摩尔投票法:找出出现次数超过 n/3 的元素

文章目录 问题描述关键洞察算法原理Java 实现算法演示投票阶段验证阶段 复杂度分析算法关键点通用化公式实际应用场景边界情况处理总结 标签&#xff1a;LeetCode 169, 摩尔投票法, 多数元素, 算法扩展, 数组处理 在解决多数元素问题时&#xff0c;我们学习了经典的摩尔投票法处…

Git:现代软件开发的基石——原理、实践与行业智慧·优雅草卓伊凡

Git&#xff1a;现代软件开发的基石——原理、实践与行业智慧优雅草卓伊凡 一、Git的本质与核心原理 1. 技术定义 Git是一个分布式版本控制系统&#xff08;DVCS&#xff09;&#xff0c;由Linus Torvalds在2005年为管理Linux内核开发而创建。其核心是通过快照&#xff08;Sna…

程序人生-hello’s P2P

计算机系统 大作业 题 目 程序人生-hello’s P2P 专 业 计算机与电子通信类 学   号 2023111990 班   级 23L0514 学 生 袁骋 指 导 教 师 史…

Java设计模式之设计原则

Java设计模式 Java设计模式主要原则是开闭原则&#xff0c;即对扩展开放&#xff0c;对修改关闭。由此衍生出5大原则&#xff1a;单一职责原则&#xff0c;里式替换原则&#xff0c;迪米特原则&#xff0c;接口隔离职责&#xff0c;依赖倒置原则。1、开闭原则 开闭原则&#x…

使用 ssld 提取CMS 签名并重签名

拿SpringBoard的cms签名和entitlements.xml&#xff0c;对tihook.dylib进行重签名 工具来源&#xff1a;https://github.com/eksenior/ssld

WebFuture:测试邮件发送失败

问题描述&#xff1a;测试邮件发送失败 问题分析&#xff1a; 查看报错是模拟发送邮件请将systemsettings.json中的EnabledMail设为false&#xff01; 解决方案&#xff1a; 网站根目录找到Configuration&#xff0c;如下图所示&#xff0c;将systemsettings.json中的Enabled…

LiveNVR 直播流拉转:Onvif/RTSP/RTMP/FLV/HLS 支持海康宇视天地 SDK 接入-视频广场页面集成与视频播放说明

LiveNVR直播流拉转&#xff1a;Onvif/RTSP/RTMP/FLV/HLS支持海康宇视天地SDK接入-视频广场页面集成与视频播放说明 一、视频页面集成1.1 关闭接口鉴权1.2 视频广场页面集成1.2.1 隐藏菜单栏1.2.2 隐藏播放页面分享链接 1.3 其它页面集成 二、播放分享页面集成2.1 获取 iframe 代…

12. CSS 布局与样式技巧

在前端开发中&#xff0c;CSS 是控制页面样式和布局的核心技术。本文总结了 CSS 布局中的关键概念和实用技巧&#xff0c;包括 overflow 属性、背景图片处理、精灵图技术、display 属性、浮动布局以及清除浮动的方法。 一、overflow 属性详解 overflow 属性用于控制当元素内容…

OpenCV---Canny边缘检测

一、基本概念与核心作用 Canny边缘检测是计算机视觉中最经典的边缘检测算法之一&#xff0c;由John Canny于1986年提出。其核心目标是在噪声图像中提取精确、单像素宽、连续的边缘&#xff0c;广泛应用于&#xff1a; 目标检测预处理&#xff08;如Robomaster中灯条、装甲板的…

提效-点击跳转到源码

1、localhost项目 例如【鲸岛】这个中台项目启动地址是localhost。 使用chrome中的【click-to-react-component 】扩展&#xff0c; alt 鼠标左键 选择dom后跳转到对应文件。 click-to-react-component的原理&#xff08;来自ai&#xff09; click-to-react-component 的工作…

FeignClient发送https请求时的证书验证原理分析

背景 微服务之间存在调用关系&#xff0c;且部署为 SSL 协议时&#xff0c;Feignt 请求报异常&#xff1a; Caused by: javax.net.ssl.SSLHandshakeException: PKIX path building failed: sun.security.provider.certpath.SunCertPathBuilderException: unable to find vali…