Java当中的深拷贝和浅拷贝

news2024/11/24 22:33:34

文章目录

  • 一、前提
  • 二、浅拷贝
    • 1. BeanUtils实现浅拷贝
  • 三、深拷贝
    • 1. 实现Cloneable接口并重写clone()方法:
    • 2. 使用序列化与反序列化:

一、前提

在计算机的内存中,每个对象都被赋予一个地址,该地址指向对象在内存中存储的位置。当我们使用一个变量来引用一个对象时,实际上是将该对象的地址赋值给该变量。因此,当我们将一个对象复制到另一个变量中时,实际上是将对象的地址复制给了这个变量。

Java当中的深拷贝和浅拷贝

二、浅拷贝

浅拷贝是一种复制对象到另一个变量的操作,它仅复制对象的地址而非对象本身。换句话说,原始对象和复制对象实际上共享同一个内存地址。因此,若我们修改复制对象中的属性或元素,原始对象中对应的属性或元素也会随之改变。

1. BeanUtils实现浅拷贝

🎏下面是一个使用BeanUtils实现深拷贝不生效的示例:🎏

首先,我们定义两个类,分别是Person和Address:

public class Person {
    private String name;
    private int age;
    private Address address;

    // getters and setters
}

public class Address {
    private String city;
    private String street;

    // getters and setters
}

然后,我们创建一个Person对象,并设置其属性:

Person person1 = new Person();
person1.setName("John");
person1.setAge(25);

Address address1 = new Address();
address1.setCity("New York");
address1.setStreet("123 Main St");

person1.setAddress(address1);

接下来,我们使用BeanUtils进行深拷贝:

Person person2 = new Person();

try {
    BeanUtils.copyProperties(person2, person1);
} catch (IllegalAccessException | InvocationTargetException e) {
    e.printStackTrace();
}

现在,我们修改person1的属性值,观察person2的属性值是否发生变化:

person1.setName("Mike");
person1.getAddress().setCity("Los Angeles");

最后,我们打印person1和person2的属性值,检查深拷贝是否生效:

System.out.println(person1.getName()); // Output: Mike
System.out.println(person1.getAddress().getCity()); // Output: Los Angeles

System.out.println(person2.getName()); // Output: Mike
System.out.println(person2.getAddress().getCity()); // Output: Los Angeles

从输出结果可以看出,尽管我们使用BeanUtils进行了拷贝,但person1和person2的属性值仍然是相同的,修改其中一个对象的属性值会同时影响另一个对象的属性值。这说明使用BeanUtils进行拷贝时,并没有进行深拷贝,而是进行了浅拷贝。

三、深拷贝

深拷贝是一种复制对象及其所有子对象到另一个变量的操作。它会创建一个全新的对象,并将原始对象中的所有属性或元素都复制到新的对象中。因此,若我们修改复制对象中的属性或元素,原始对象中对应的属性或元素不会受到任何影响。

在Java中实现深拷贝有以下几种方法:

1. 实现Cloneable接口并重写clone()方法:

✨在需要进行深拷贝的类中,实现Cloneable接口并重写clone()方法,在clone()方法中对引用类型进行逐个拷贝。注意,被拷贝的引用类型也需要实现Cloneable接口并重写clone()方法。

下面是一个使用Cloneable接口实现深拷贝的示例代码:

class Person implements Cloneable {
    private String name;
    private int age;

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

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

    public String getName() {
        return name;
    }

    public int getAge() {
        return age;
    }

    public void setName(String name) {
        this.name = name;
    }

    public void setAge(int age) {
        this.age = age;
    }
}

public class DeepCopyDemo {
    public static void main(String[] args) {
        Person person1 = new Person("John", 25);

        try {
            // 使用clone()方法进行深拷贝
            Person person2 = (Person) person1.clone();

            System.out.println("person1: " + person1.getName() + ", " + person1.getAge());
            System.out.println("person2: " + person2.getName() + ", " + person2.getAge());

            person2.setName("Tom");
            person2.setAge(30);

            System.out.println("After modifying person2:");
            System.out.println("person1: " + person1.getName() + ", " + person1.getAge());
            System.out.println("person2: " + person2.getName() + ", " + person2.getAge());
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
    }
}

输出结果:

person1: John, 25
person2: John, 25
After modifying person2:
person1: John, 25
person2: Tom, 30

从输出结果可以看出,person2是通过深拷贝得到的新对象,修改person2的属性不会影响到person1。这说明使用Cloneable接口和clone()方法可以实现深拷贝。需要注意的是,被拷贝的类及其引用类型属性都需要实现Cloneable接口并重写clone()方法。

2. 使用序列化与反序列化:

🎯将对象序列化为字节流,然后再反序列化为新的对象。这种方法需要被拷贝的类实现Serializable接口。🎯

下面是一个使用序列化与反序列化实现深拷贝的示例代码:

import java.io.*;

class Person implements Serializable {
    private String name;
    private int age;

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public int getAge() {
        return age;
    }

    public void setName(String name) {
        this.name = name;
    }

    public void setAge(int age) {
        this.age = age;
    }
}

public class DeepCopyDemo {
    public static void main(String[] args) {
        Person person1 = new Person("John", 25);

        try {
            // 序列化
            ByteArrayOutputStream bos = new ByteArrayOutputStream();
            ObjectOutputStream oos = new ObjectOutputStream(bos);
            oos.writeObject(person1);

            // 反序列化
            ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
            ObjectInputStream ois = new ObjectInputStream(bis);
            Person person2 = (Person) ois.readObject();

            System.out.println("person1: " + person1.getName() + ", " + person1.getAge());
            System.out.println("person2: " + person2.getName() + ", " + person2.getAge());

            person2.setName("Tom");
            person2.setAge(30);

            System.out.println("After modifying person2:");
            System.out.println("person1: " + person1.getName() + ", " + person1.getAge());
            System.out.println("person2: " + person2.getName() + ", " + person2.getAge());
        } catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

输出结果:

person1: John, 25
person2: John, 25
After modifying person2:
person1: John, 25
person2: Tom, 30

从输出结果可以看出,person2是通过反序列化得到的新对象,修改person2的属性不会影响到person1。这说明使用序列化与反序列化可以实现深拷贝。需要注意的是,被拷贝的类及其引用类型属性都需要实现Serializable接口。

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

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

相关文章

超详细图文教程:3DS Max 中创建低多边形游戏长剑模型

推荐: NSDT场景编辑器助你快速搭建可二次开发的3D应用场景 在此,由两部分组成的教程的第一部分中,我将向您展示如何: 对剑柄进行建模剑的护手模型剑刃建模 1. 如何制作剑柄 步骤 1 在本教程中使用正交视图。要更改视图&#x…

uniapp小程序跳转其他小程序uni.navigateToMiniProgram效果demo(整理)

放点击事件里面即可 uni.navigateToMiniProgram({appId: , //跳转的小程序的aooIdpath: pages/index/index?id123, //如果这里不填,默认是跳转到对方小程序的主页面extraData: { //需要传给对方小程序的数据data1: test},success(res) {// 打开成功} })

Leetcode每日一题:42. 接雨水(2023.7.23 C++)

目录 42. 接雨水 题目描述: 实现代码与解析: 单调栈 原理思路: 动态规划 原理思路: 42. 接雨水 原题链接: 42. 接雨水 题目描述: 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,…

SVTR: 使用单一视觉模型进行场景文本识别【论文翻译】

🥇 版权: 本文由【墨理学AI】原创首发、各位读者大大、敬请查阅、感谢三连 🎉 声明: 作为全网 AI 领域 干货最多的博主之一,❤️ 不负光阴不负卿 ❤️ 文章目录 SVTR: Scene Text Recognition with a Single Visual ModelAbstract1 Introduct…

Ansible部署与实施PlayBook

Ansible部署与实施PlayBook 计算机名称IP角色workstation192.168.182.130管理servera192.168.182.131被管理serverb192.168.182.132被管理serverc192.168.182.133被管理 部署 控制节点 官方文档 Ansible易于安装。只需要安装到要运行它的一个或多个控制节点上。由Ansbile管…

【事业单位-语言理解】中心理解01

【事业单位-语言理解】中心理解01 一、中心理解1.1 转折关系1.2 因果关系1.3必要条件关系 二、总结 一、中心理解 1.1 转折关系 转折之后是中心意思 转折在分述句子中,就没有那么重要 1.2 因果关系 一方面另一方面起到的是分述的作用,一般不要过多…

python解析markdown

python解析markdown 1、安装markdown模块2、python解析markdown2.1 QtWebEngineWidgets显示网页2.1.1 网页加载2.1.2 网页加载错误 2.2 Python-Markdown 模块库2.3 QTextDocument、QTextBrowser、QTextEdit 3、案例Demo3.1 先上图3.2 代码文件 1、安装markdown模块 pip instal…

ElasticSearch学习--操作

目录 索引库操作 mapping映射 总结 创建索引库 查询删除修改索引库 总结 文档操作 添加 查询,删除 修改文档 总结 RestClient操作索引库 初始化JavaRestClient 创建索引库​编辑 删除索引库,判断索引库是否存在 总结 操作文档 新增文档 查询文…

JAVA SE -- 第九天

(全部来自“韩顺平教育”) 一、类变量个类方法 (一)类变量 1、基本介绍 类变量也叫静态变量/静态属性,是该类的所有对象共享的变量,任何一个该类的对象去访问它时,取到的都是相同的值&#…

2023年的深度学习入门指南(18) - 将LLaMA2跑起来

2023年的深度学习入门指南(18) - 将LLaMA2跑起来 之前我们说到过,在GPT 3之后,大模型就很少有开源的了。其中,最为典型的开源支持者就是Meta公司的研究团队。年初他们发布的LLaMA基本上是各家开源模型的主要参考对象。不过,LLaMA…

Python 算法基础篇之集合和字典:创建、访问、添加和删除元素

Python 算法基础篇之集合和字典:创建、访问、添加和删除元素 引言 1. 集合的概念和创建2. 集合的访问3. 集合的添加和删除元素 a ) 添加元素 b ) 删除元素 4. 字典的概念和创建5. 字典的访问6. 字典的添加和删除元素 a ) 添加元素 b ) 删除元素 总结 引言 集合和字…

SAP从放弃到入门系列之批次派生-Batch Derivation-Part2

文章目录 一、派生的类型1.1 静态派生1.2 动态派生 二、派生的方向 通过批次派生的基本配置和简单功能的介绍,大家应该对批次派生有一个基本的了解,这篇文章从批次派生的类型和批次派生的方向两个维度更深入的聊一下它的功能。 一、派生的类型 派生的类…

OPTEE之sonarlint静态代码分析实战一——optee_os

ATF(TF-A)/OPTEE之静态代码分析汇总 一、OPTEE源码下载及分析 对OPTEE实施soanrlint静态代码扫描之前,先到官方网站下载源码。官方网站位于github,网址OP-TEE GitHub。 其中我们重点关注optee_os和optee_client。此页面下的optee_linuxdriver已废弃,该部分最终会编…

数据结构:二叉树遍历

概述 二叉树的遍历是指按照某条搜索路径访问二叉树中的每个结点,使得每个结点均被访问一次,而且仅被访问一次。二叉树的遍历方式主要有:先序遍历、中序遍历、后序遍历、层次遍历。先序、中序、后序其实值得是父节点被访问的次序。若在遍历过…

一.MySQL的主从复制

目录 一.MySQL的主从复制 1.2主从复制的工作过程和原理 1.2.1主从复制工作过程为两日志和三线程 ​编辑 1.2.2主从复制的工作原理 1.3主从复制延迟的原因 1.4主从复制的架构 1.5.MySQL四种同步方式 1.5.1异步复制(Async Replication) 1.5.2 同步复制(Sync Re…

Linux内核子系统--进程管理剖析

Linux 是一个计算需求不断变化的非常动态的系统。 Linux 计算需求的表示以进程的公共抽象为中心,进程可以是短期的(从命令行执行的命令)或长期的(网络服务)。因此,进程的总体管理及其调度非常重要。 在用户…

EMC学习笔记(十八)滤波器设计

滤波器设计 1.标准要求2.设计理论2.1 滤波器电路设计过程2.2 插入损耗定义2.3 原始噪声测量2.4 插入损耗计算2.5 滤波失配原则2.6 滤波拓扑选择2.7 滤波参数计算2.8 滤波参数确定 Tips:学习资料来自网络,仅供学习使用。 EMI滤波器设计(汽车电…

JVM系列(9)——调优初体验

学习这篇文章之前,要了解: JVM系列(2)——垃圾回收 JVM系列(3)——内存分配与回收策略 先了解概念: 吞吐量:用户执行时间/(用户执行时间垃圾回收时间);就是干正经事的时间…

C++语法(25)--- 异常与智能指针

C语法(24) C11_哈里沃克的博客-CSDN博客https://blog.csdn.net/m0_63488627/article/details/131054426?spm1001.2014.3001.5501 1.异常 try { // 保护的标识代码 }catch( ExceptionName e1 ) { // catch 块 }catch( ExceptionName e2 ) { // catch 块…

RT-Thread qemu mps2-an385 bsp 移植制作 :BSP 制作篇

下载 V2M-MPS2_CMx_BSP mps2 的资料很少,所以唯一能下载的是 ARM 官方的 V2M-MPS2_CMx_BSP,下载地址为: https://keilpack.azureedge.net/pack/Keil.V2M-MPS2_CMx_BSP.1.8.0.pack 其实这是个 Keil MDK5 的 Pack 包,安装后&#x…