深拷贝和浅拷贝 深克隆和浅克隆

深拷贝和浅拷贝是用来描述对象或者对象数组这种引用数据类型的复制场景的。
浅拷贝,就是只复制某个对象的指针,而不复制对象本身。
这种复制方式意味着两个引用指针指向被复制对象的同一块内存地址。
深拷贝,会完全创建一个一模一样的新对象,新对象和老对象不共享内存,也就意味着对新对象的修
改不会影响老对象的值。
在Java里面,无论是深拷贝还是浅拷贝,都需要通过实现Cloneable接口,并实现clone()方法。
然后我们可以在clone()方法里面实现浅拷贝或者深拷贝的逻辑。
实现深拷贝的方法有很多,比如通过序列化的方式实现,也就是把一个对象先序列化一遍,然后再反序列化回来,就会得到一个完整的新对象。
在clone()方法里面重写克隆逻辑,也就是对克隆对象内部的引用变量再进行一次克隆。

深克隆和浅克隆 深拷贝和浅拷贝

深克隆(Deep Clone)和浅克隆(Shallow Clone)是对象复制的两种不同方式。
深克隆是指在克隆对象时,将原始对象及其所有引用类型成员对象的内容都复制一份到新的对象中,新旧对象之间不存在共享的引用关系。也就是说,深克隆会递归地复制对象的所有层级,使得新创建的对象与原始对象完全独立,修改其中一个对象不会影响另一个对象。
浅克隆是指在克隆对象时,只复制对象本身,而不复制对象所包含的引用类型成员对象。新创建的对象与原始对象共享部分或全部引用类型成员对象。这意味着,修改其中一个对象的引用类型成员对象会影响另一个对象。
深拷贝(Deep Copy)和浅拷贝(Shallow Copy)是数据复制的概念,与克隆概念类似。
深拷贝是指在拷贝数据时,将原始数据及其所有嵌套的数据结构都复制一份,新旧数据之间不存在共享的引用关系。
浅拷贝是指在拷贝数据时,只复制数据本身,而不复制嵌套的数据结构。新创建的数据与原始数据共享部分或全部嵌套的数据结构。
所以,深克隆和深拷贝是指完全复制对象及其嵌套的所有成员,而浅克隆和浅拷贝是指仅复制对象本身或数据本身,不涉及嵌套成员的复制。它们并不是完全相同的概念,但在某些情况下可以有类似的效果,取决于对象的结构和属性的可变性。

浅克隆:被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象。
深克隆:除去那些引用其他对象的变量,被复制对象的所有变量都含有与原来的对象相同的值。那些引用其他对象的变量将指向被复制过的新对象,而不再是原有的那些被引用的对象。换言之,深复制把要复制的对象所引用的对象都复制了一遍

深克隆和浅克隆

实现浅克隆的常用API有以下3种:

1、工具类BeanUtils和PropertyUtils,BeanUtils是Spring提供的,PropertyUtils是Apache的commons包中
提供的。
2、实现Clonenable接口
3、Arrays的copyOf()方法
在Java中,实现浅克隆可以使用以下几种方式:

  1. Object类的clone()方法:Object类是所有类的父类,它提供了一个clone()方法用于克隆对象。需要注意的是,Object类的clone()方法是浅克隆,需要在需要克隆的类中重写该方法来实现浅拷贝。例如:
=class MyClass implements Cloneable {private int num;public void setNum(int num) {this.num = num;}public int getNum() {return num;}@Overridepublic Object clone() throws CloneNotSupportedException {return super.clone();}
}// 测试代码
MyClass obj1 = new MyClass();
obj1.setNum(10);try {MyClass obj2 = (MyClass) obj1.clone();System.out.println(obj2.getNum()); // 输出: 10
} catch (CloneNotSupportedException e) {e.printStackTrace();
}
  1. Cloneable接口:Cloneable接口是一个标记接口,表示该类能够被克隆。需要实现Cloneable接口,并重写clone()方法来实现浅克隆。注意,如果没有实现Cloneable接口,调用clone()方法将会抛出CloneNotSupportedException异常。
  2. 使用拷贝构造函数:在类中定义一个拷贝构造函数,用于创建一个新对象并将原始对象的属性值复制给新对象。这种方式适用于简单对象的复制,但对于嵌套对象则需要额外处理。
  3. 使用BeanUtils类:Apache Commons项目中的BeanUtils类提供了一个copyProperties()方法,可以将一个对象的属性值复制到另一个对象中。这是一种方便快捷的浅拷贝方式,但需要导入相应的jar包。
    需要注意的是,浅克隆只能复制对象本身,对于对象内部的引用类型成员对象,则只是复制了引用,两个对象仍然共享同一个成员对象。如果需要实现深克隆,则需要考虑使用深拷贝的方式来复制嵌套的成员对象。

深克隆常用的API有以下5种

1、重写clone()方法
2、序列化
3、Apache commons工具包中的SerializationUtils的clone()方法
4、JSON工具类
5、通过构造方法手动new对象

在Java中,实现深克隆可以使用以下几种常用的API和方式:

  1. 通过序列化和反序列化实现深克隆:利用对象的序列化和反序列化可以实现深克隆。首先将对象写入输出流,然后再从输入流中读取对象,这样就可以得到一个新的对象,而不是原始对象的引用。示例代码如下:
import java.io.*;class MyClass implements Serializable {private int num;public void setNum(int num) {this.num = num;}public int getNum() {return num;}
}// 测试代码
MyClass obj1 = new MyClass();
obj1.setNum(10);try {// 将对象写入输出流ByteArrayOutputStream baos = new ByteArrayOutputStream();ObjectOutputStream oos = new ObjectOutputStream(baos);oos.writeObject(obj1);// 从输入流中读取对象ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());ObjectInputStream ois = new ObjectInputStream(bais);MyClass obj2 = (MyClass) ois.readObject();System.out.println(obj2.getNum()); // 输出: 10
} catch (IOException | ClassNotFoundException e) {e.printStackTrace();
}
1. 使用Apache Commons库中的SerializationUtils类:Apache Commons项目提供了SerializationUtils类,它提供了一个clone()方法来实现深克隆。
javaCopy Codeimport org.apache.commons.lang3.SerializationUtils;class MyClass implements Serializable {private int num;public void setNum(int num) {this.num = num;}public int getNum() {return num;}
}// 测试代码
MyClass obj1 = new MyClass();
obj1.setNum(10);MyClass obj2 = SerializationUtils.clone(obj1);
System.out.println(obj2.getNum()); // 输出: 10
  1. 使用第三方库,如Gson、Jackson等:一些第三方库提供了对象的序列化和反序列化功能,也可以用来实现深克隆。
    需要注意的是,使用序列化和反序列化方式进行深克隆时,需要确保对象及其成员对象都实现了Serializable接口,否则可能会导致克隆失败。另外,深克隆的性能相对较低,因为涉及对象序列化和反序列化的操作。

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

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

相关文章

一步解决 java.io.FileNotFoundException: 找不到文件异常

1.问题描述 java.io.FileNotFoundException: C:\Users\Administrator\AppData\Local\Temp\localhost\uploads\image\20231206\2843cb16-9654-4e52-a757-76e3ca1f80ff.png (系统找不到指定的路径。) 2.原因分析 文件路径中的文件目录不存在 3.解决方案 方案一:如果…

ArrayList扩容机制

1.初始容量 在源码中,ArrayList根据传递参数的不同有三种不同的构造方式 传递一个整数 新创建集合大小等于传递的整数大小空参构造 新创建集合大小为空传递集合 新创建集合大小等于传递的参数集合大小 源码如下: //1.传递一个整数 public ArrayList…

行业研究:2023年中国游戏陪玩行业市场现状分析

近年来随着我国游戏行业的不断发展,我国游戏用户规模也是随着稳步上升,给游戏陪玩行业带来了稳定的用户基础。在用户规模增长的同时,随着经济、文化的快速发展,我国娱乐技能社交也随之逐渐兴起。而作为我国娱乐技能社交比重较大的…

java中乐观锁与悲观锁的区别是什么?

乐观锁和悲观锁是两种处理并发访问的不同策略。 悲观锁: 特点: 假设会发生并发冲突,因此在整个操作过程中都持有锁,防止其他线程访问。实现方式: 使用传统的加锁机制,如 synchronized 关键字或 Lock 接口…

Apache POI操作excel

使用Apache POI 引入坐标 <dependency><groupId>org.apache.poi</groupId><artifactId>poi</artifactId><version>3.16</version> </dependency> <dependency><groupId>org.apache.poi</groupId><artifa…

Vue学习计划-Vue2--Vue核心(七)生命周期

抛出问题&#xff1a;一进入页面就开启一个定时器&#xff0c;每隔1秒count就加1&#xff0c;如何实现 示例&#xff1a; <body> <div id"app">{{ n }}<button click"add">执行</button> </div><script>let vm new …

一文搞懂Git版本控制系统

1. Git简介 当涉及到软件开发或协作时&#xff0c;版本管理是一个不可或缺的概念。无论你是一个独立开发者还是一个团队成员&#xff0c;都会遇到需要跟踪和管理代码变更的情况。这时候&#xff0c;Git作为一个强大而流行的版本控制系统就发挥着重要的作用。 Git&#xff08;读…

玩转大数据9:机器学习在大数据分析中的应用

1. 引言 在大数据时代&#xff0c;机器学习在大数据分析中扮演着至关重要的角色。本文介绍机器学习在大数据分析中的重要性和应用场景&#xff0c;并探讨Java中可用的机器学习库和框架。 2. 机器学习的基本概念和算法 机器学习是当今人工智能领域的一个关键分支&#xff0c;…

区分物理端口与软件端口概念:以交换机端口和Linux系统中的端口为例

文章目录 交换机端口和Linux系统中的端口有什么区别&#xff1f;1. 交换机的端口2. Linux系统中的端口因此&#xff0c;尽管两者都被称为"端口"&#xff0c;但它们代表的含义和用途是完全不同的。 交换机端口和Linux系统中的端口有什么区别&#xff1f; 虽然都被称为…

智慧园区可视化综合管理平台建设方案,智能化、数字化才是关键

园区作为城市的基本单元&#xff0c;是经济发展的重要载体。随着我国经济的快速发展&#xff0c;各类工业园区、办公园区等园区的规划建设也越来越多。伴随着互联网新兴技术的发展和应用&#xff0c;智慧园区已成为当今城市规划和社会发展的关注焦点&#xff0c;今天我们来介绍…

实战oj题——设计循环队列

前言&#xff1a;今天我们来实现循环队列。 各个接口的实现 创建队列&#xff1a; typedef struct {int* a;int front;int back;int k;} MyCircularQueue;我们的队列是由数组储存的&#xff0c;所以我们队列中得定义一个数组&#xff0c;front代表我们的首元素&#xff0c;ba…

腾讯云轻量应用服务器怎么安装BT宝塔面板?

腾讯云轻量应用服务器宝塔面板怎么用&#xff1f;轻量应用服务器如何安装宝塔面板&#xff1f;在镜像中选择宝塔Linux面板腾讯云专享版&#xff0c;在轻量服务器防火墙中开启8888端口号&#xff0c;然后远程连接到轻量服务器执行宝塔面板账号密码查询命令&#xff0c;最后登录和…

Gti GUI添加标签

通过Git Gui打开项目&#xff0c;通过菜单打开分支历史&#xff0c;我这里是名为"develop"的分支 选中需要打标签的commit&#xff0c;右键-Create tag即可 但貌似无法删除标签&#xff0c;只能通过git bash

卡码网语言基础课 | 20. 排队取奶茶

目录 一、 队列的基本认识 二、 队列的操作 2.1 引入头文件 2.2 创建队列 2.3 队列的常见操作 三、 解题 通过本次练习&#xff0c;将会学习到以下C知识点&#xff1a; 队列的基本概念&#xff08;队头、队尾&#xff09;和特点&#xff08;先入先出&#xff09;入队、出队…

Android : Xui- RecyclerView+BannerLayout 轮播图简单应用

实例图&#xff1a; 1.引用XUI http://t.csdnimg.cn/Wb4KR 2.创建显示图片布局 banner_item.xml <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas.android.com/apk/res/android"xmlns:app"…

【CentOS8】使用 Tomcat 部署 Java Web 项目(使用 sdkman)

文章目录 配置 Tomcat将 Tomcat 启动命令设置为 Linux 自定义服务给 Tomcat 设置管理员账号密码IDEA 打包 Java web 项目 我是使用 sdkman 下载的 jdk 和 tomcat&#xff0c;所以接下来的部署配置都是在 sdkman 构建的环境的。想要知道如何下载 sdkman 可以看看这篇文章 —…

cv2.error: OpenCV(4.7.0)

运行hsv脚本报错&#xff1a; cv2.error: OpenCV(4.7.0) D:\a\opencv-python\opencv-python\opencv\modules\imgproc\src\color.cpp:182: error: (-215:Assertion failed) !_src.empty() in function cv::cvtColor 解决方案&#xff1a; 这个错误信息是在使用OpenCV的cvtColor函…

计网实验7

解决&#xff1a;路由器用rip连接&#xff0c;主机通过域名访问&#xff0c;主机之间发送电子邮件 实验步骤 1.搞好部件 2.配好两台主机的ip,掩码&#xff0c;网关 3.连接一下两台主机&#xff0c;由于两台路由器没有连接&#xff0c;所以两台主机也无法连通&#xff0c;丢包率…

Flink优化——数据倾斜(二)

目录 数据倾斜 判断是否存在数据倾斜 数据倾斜的解决 KeyBy之前发生数据倾斜 KeyBy之后发生的数据倾斜 聚合操作存在数据倾斜 窗口聚合操作存在数据倾斜 数据倾斜 判断是否存在数据倾斜 相同 Task 的多个 Subtask 中&#xff0c;个别 Subtask 接收到的数据量明显大于其…

如何快速移植(从STM32F103到STM32F407)

最近用到F4的地方比较多&#xff0c;网上代码还是F1多一些&#xff0c;便需要移植代码&#xff0c;如何快速移植代码呢&#xff1f; 看下面这篇文章 外设 首先就是STM32的外设了。 STM32F407ZGT6的基本外设 STM32F407ZGT6 作为 MCU&#xff0c;该芯片是 STM32F407 里面配置…