Java中的深拷贝和浅拷贝

目录

🍎引出拷贝

🍎浅拷贝

🍎深拷贝

🍎总结


引出拷贝

现在有一个学生类和书包类,在学生类中有引用类型的书包变量:

class SchoolBag {
    private String brand;   //书包的品牌
    private int size;   //书包的尺寸
    //getter、setter略

    public SchoolBag(String brand, int size) {
        this.brand = brand;
        this.size = size;
    }

    @Override
    public String toString() {
        return "SchoolBag{" +
                "brand='" + brand + '\'' +
                ", size=" + size +
                '}';
    }
}

class Student {
    private String name;
    private int age;
    private SchoolBag schoolBag;

    public Student(String name, int age, SchoolBag schoolBag) {
        this.name = name;
        this.age = age;
        this.schoolBag = schoolBag;
    }

    //getter、setter略

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", schoolBag=" + schoolBag +
                '}';
    }
}

创建一个学生对象,现在想在内存中拷贝一份和这个学生对象指向的对象相同的对象,那么应该怎么去做呢?

可能我们会像下边这样写:

...
Student stu1 = new Student("张华",18,new SchoolBag("小米",4));
//直接创建一个新的引用指向这个stu1指向的对象
Student stu2 = stu1;

这显然是错误的,这从内存的分布上就可以知道,stu1与stu2虽然是两个不同的引用变量,但是指向的是堆内存中的同一块空间或者说指向的是同一个对象,并没有真正实现克隆的效果。它们两个引用和引用对象在内存中的关系如图

那么要怎么实现对象内存的拷贝呢?


浅拷贝

 我们可以通过使用Java中的clone方法,该clone方法是Object类中定义的,因此我们只需要在自己的类中重写这个方法,就可以实现对象的拷贝。但是clone方法的使用,并不只是单纯的重写就可以实现效果的,下面我们依次来看:

1.待克隆对象所在类重写clone方法

class Student {
    private String name;
    private int age;
    private SchoolBag schoolBag;

    public Student(String name, int age, SchoolBag schoolBag) {
        this.name = name;
        this.age = age;
        this.schoolBag = schoolBag;
    }

    //getter、setter略

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", schoolBag=" + schoolBag +
                '}';
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

调用重写的clone方法,克隆对象,会发现,报错了:

哦😂,原来还得让待克隆对象所在类实现Cloneable接口,但是当我们查看这个接口的内容时,却发现这是一个空接口:

为什么是一个空接口呢?

这里可以记作它是一个标记接口,实现Cloneable接口的类被标记为可以被clone的类。

现在我们让Student类实现Cloneable接口再试试看能不能克隆成功。

2.实现Cloneable接口

class Student implements Cloneable{
    private String name;
    private int age;
    private SchoolBag schoolBag;

    public Student(String name, int age, SchoolBag schoolBag) {
        this.name = name;
        this.age = age;
        this.schoolBag = schoolBag;
    }

    //getter、setter略

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", schoolBag=" + schoolBag +
                '}';
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

现在再次尝试clone,嗯...又报错了......

原来时异常,我们把异常抛给JVM一下:

 啊,还报错...

原来是clone方法的返回类型时Object:

 我们进行向下转型,然后进行克隆输出会发现克隆成功了

 现在克隆出来的对象以及原型在内存中因该是这样分布的:

当我们修改stu1对象中的schoolbag成员变量后再进行对象的输出,会发现stuClone对象的schoolbag中的值也发生了变化,这是为什么?????

 这是因为clone出来的stuClone对象和stu1对象的引用成员变量schoolbag指向的还是同一片内存空间,二者还是共享这一片内存空间。
 

因此我们可以得到何为浅拷贝?

当引用变量指向的对象中还包含引用变量时,如果只对当前的外部的引用变量进行clone,克隆出来对象的基本数据类型都会被正常拷贝;而内部的引用变量只是将引用的值clone给了另一个clone对象的引用,因此无法在真正意义上实现对象的拷贝,下面我们来解决这种clone后仍存在成员变量有内存共享的问题。


深拷贝

既然clone方法能够实现对对象的拷贝,那么我们在引用变量内部的引用变量身上再使用一次clone方法不就可以解决这种浅拷贝问题了吗?

现在我们进行上述操作,让SchoolBag类重写clone方法并且实现标记接口Cloneable:

class SchoolBag implements Cloneable{
    private String brand;   //书包的品牌
    private int size;   //书包的尺寸
    //getter、setter略

    public SchoolBag(String brand, int size) {
        this.brand = brand;
        this.size = size;
    }

    @Override
    public String toString() {
        return "SchoolBag{" +
                "brand='" + brand + '\'' +
                ", size=" + size +
                '}';
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

在对Student对象进行克隆是也对其中的引用成员变量进行克隆,我们只需要在Student类的clone方法中这样写就可以了:

class Student {
    ...
    @Override
    protected Object clone() throws CloneNotSupportedException {
        Student stuClone = (Student)super.clone();
        SchoolBag schoolbagClone = (SchoolBag) this.schoolBag.clone();
        stuClone.setSchoolBag(schoolbagClone);
        return stuClone;
    }
    ...
}

现在我们在进行stu对象的schoolbag修改后的输出,会发现stu1对象的修改已经不会影响stuClone对象了,二者之间已经没有共享的存储空间了:

 此时这两个对象的内存布局图如下:

 因此我们可以得到何为深拷贝?

深拷贝进行后,进行拷贝的对象与拷贝出来的对象之间没有共享的一片存储空间,二者独立存在,对其中任何一方的修改都不会影响另外一方。


总结

拷贝实现方法clone的使用步骤以及注意事项:

  1. 实现Cloneable标记接口重写Object类中的clone方法
  2. 捕获处理或抛出使用clone方法带来的异常
  3. 将克隆猴返回的Object类型的对象向下转型为克隆类型
  4. 如果克隆对象的内部还有引用类型,则需要对内部的引用类型也按照上述1,2,3步骤在自己的clone方法中进行克隆返回。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/3809.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

7.网络爬虫—正则表达式详讲

7.网络爬虫—正则表达式详讲与实战Python 正则表达式re.match() 函数re.search方法re.match与re.search的区别re.compile 函数检索和替换检索:替换:findallre.finditerre.split正则表达式模式常见的字符类正则模式正则表达式模式量词正则表达式举例前言&…

2022财报逆转,有赞穿透迷雾实现突破

2022年,商家经营面临困难。但在一些第三方服务商的帮助下,也有商家取得了逆势增长。 2023年3月23日,有赞发布2022年业绩报告,它帮助许多商家稳住了一整年的经营。2022年,有赞门店SaaS业务的GMV达到425亿元&#xff0c…

24万字智慧城市顶层设计及智慧应用解决方案

本资料来源公开网络,仅供个人学习,请勿商用,如有侵权请联系删除。部分资料内容: 4.8 机房消防系统 4.8.1消防系统概况 根据本工程机房消防系统的特殊要求,整个消防系统由火灾报警系统、消防联动系统和气体灭火系统三部…

常见的嵌入式微处理器(Micro Processor Unit,MPU)

嵌入式微处理器是由通用计算机中的CPU演变而来的。它的特征是具有32位以上的处理器,具有较高的性能,当然其价格也相应较高。但与计算机处理器不同的是,在实际嵌入式应用中,只保留和嵌入式应用紧密相关的功能硬件,去除了…

医院陪诊系统源码,可以提供新的就医方式

随着人们生活水平的提高和医疗服务的进步,越来越多的人们开始注重家庭健康和医疗保健。在这个背景下,陪护系统和医院陪诊系统应运而生,成为了现代医疗服务领域中的重要组成部分。 陪护系统是一种为患者提供家庭养护服务的机构,它…

“蓝桥杯”递推和递归(一)——取数位

1. 算法简介 递推和递归虽然叫法不同,但它们的基本思想是一致的,在很多程序中,这两种算法可以通用,不同的是递推法效率更高,递归法更方便阅读。 (1)递推法 递推法是一种重要的数学方法&#…

【PC自动化测试-4】inspect.exe 详解

1,inspect.exe图解" 检查 "窗口有几个主要部分:● 标题栏。 显示" 检查 HWND (窗口句柄) 。● 菜单栏。 提供对 检查功能 的访问权限。● 工具 栏。 提供对 检查功能 的访问权限。● 树视图。 将 UI 元素的层次结构呈现为树视图控件&…

【超好懂的比赛题解】暨南大学2023东软教育杯ACM校赛个人题解

title : 暨南大学2023东软教育杯ACM校赛 题解 tags : ACM,练习记录 date : 2023-3-26 author : Linno 文章目录暨南大学2023东软教育杯ACM校赛 题解A-小王的魔法B-苏神的遗憾C-神父的碟D-基站建设E-小王的数字F-Uziの真身G-电子围棋H-二分大法I-丁真的小马朋友们J-单车运营K-超…

JavaScript实现列表分页(小白版)

组件用惯了,突然叫你用纯cssJavaScript写一个分页,顿时就慌了。久久没有接触js了,不知道咋写了。本文章也是借与参考做的一个demo案例,小白看了都会的那种。咱们就以ul列表为例进行分页: 首先模拟的数据列表是这样的&a…

变量的理论分布模型

二项分布 定义 对立事件的总体分布,称为二项分布。 例如,一个群体只有男和女,现在进行n次随机抽样调查,随机抽样男出现的次数可能是0,1,2,3,4,…,n, 这种类…

网络安全实战从 0 到 1 彻底掌握 XXE

0x01 什么是 XXE个人认为,XXE 可以归结为一句话:构造恶意 DTD介绍 XXE 之前,我先来说一下普通的 XML 注入,这个的利用面比较狭窄,如果有的话应该也是逻辑漏洞。既然能插入 XML 代码,那我们肯定不能善罢甘休…

ROS Cartographer--Algorithm

ROS Cartographer–Algorithm 原文:Algorithm walkthrough for tuning 论文地址(Google Search):Real-Time Loop Closure in 2D LIDAR SLAM ROS Cartographer的完整参考文件:Cartographer ROS Integration 概述 本地SLAM通常由前端和后端…

Python满屏表白代码

目录 前言 爱心界面 无限弹窗 前言 人生苦短,我用Python!又是新的一周啦,本期博主给大家带来了一个全新的作品:满屏表白代码,无限弹窗版!快快收藏起来送给她吧~ 爱心界面 def Heart(): roottk.Tk…

【Linux】计算机网络1

计算机网络的背景背景:早在20世纪50年代初,美国建立的地面防空系统就是将地面的雷达和其他测量控制设备的信息通过通信线路汇集到一台中心计算机进行处理,开创了把计算机技术和通信技术相结合的尝试。20世纪60年代中期开始,出现、…

OSPF----特殊区域

目录 OSPF----特殊区域 第一大类----末梢区域(Stub Area) 完全末梢区域((Totally Stub Area) 第二大类特殊区域----非完全末梢区域(NSSA) OSPF----特殊区域 第一大类----末梢区域(Stub Area&#xff09…

动态版通讯录——“C”

各位CSDN的uu们你们好呀,今天,小雅兰的内容是动态版通讯录啦,其实之前,我就已经写过静态版的通讯录了,只是存在着一些问题,具体细节可以详细看看我的静态版通讯录,好了,话不多说&…

计算机视觉知识点(一)——交并比(IoU)及其若干改进

交并比(IoU)前言IoU公式及示意图IoU Loss缺点GIoU Loss公式及示意图缺点DIoU公式及示意图CIoU前言 目标检测是一个常见的计算机视觉任务,在目标检测任务中,交并比作为评判检测框的标准具有很重要的意义,在实际的应用中…

【百面成神】java web基础7问,你能坚持到第几问

前 言 🍉 作者简介:半旧518,长跑型选手,立志坚持写10年博客,专注于java后端 ☕专栏简介:纯手打总结面试题,自用备用 🌰 文章简介:java web最基础、重要的8道面试题 文章目…

SAP 系统中过账码or记账码

SAP中过账码和记账码是指同一个事物。 在实际业务中,记账码就是只有“借”和“贷”, 而SAP中Posting Code肩负着更多的任务: 1)界定科目类型, 2)借贷方向, 3)凭证输入时画面上的字…

运算放大器:电压比较器、电压跟随器、同相比例放大器

目录一、单限电压比较器二、滞回电压比较器三、窗口电压比较器四、正点原子直流电机驱动器电路分析实战1、电压采集电路2、电流采集电路3、过流检测电路Ⅰ、采用分压后的输入电压:Ⅱ、采用理想电压源的输入电压:Ⅲ、同相输入电压采用的是非理想电压源&am…