java中的三种拷贝方法

在Java编程中,理解深拷贝(Deep Copy)、浅拷贝(Shallow Copy)和引用拷贝(Reference Copy)是非常重要的。这三种拷贝方式涉及对象复制和内存管理。以下是对它们的详细解释:

1. 引用拷贝(Reference Copy)

引用拷贝是最简单的一种拷贝方式。它只复制对象的引用,而不复制对象本身。换句话说,原始对象和新对象共享同一个内存地址。

Person person1 = new Person("Alice");
Person person2 = person1;  // 引用拷贝person2.setName("Bob");System.out.println(person1.getName()); // 输出 "Bob"
System.out.println(person2.getName()); // 输出 "Bob"

在上述代码中,person1person2指向同一个对象,所以改变其中一个对象的属性,另一个对象的属性也会被改变。

2. 浅拷贝(Shallow Copy)

浅拷贝会创建一个新对象,但新对象中的成员变量(如果是对象)仍然是原对象的引用。浅拷贝仅复制对象的第一层属性。

可以通过实现Cloneable接口并重写clone方法来实现浅拷贝:

class Address {String city;public Address(String city) {this.city = city;}
}class Person implements Cloneable {String name;Address address;public Person(String name, Address address) {this.name = name;this.address = address;}@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();  // 浅拷贝}
}public class Main {public static void main(String[] args) throws CloneNotSupportedException {Address address = new Address("New York");Person person1 = new Person("Alice", address);Person person2 = (Person) person1.clone();person2.name = "Bob";person2.address.city = "Los Angeles";System.out.println(person1.name); // 输出 "Alice"System.out.println(person1.address.city); // 输出 "Los Angeles"System.out.println(person2.name); // 输出 "Bob"System.out.println(person2.address.city); // 输出 "Los Angeles"}
}

在上述代码中,person1person2拥有不同的name属性,但是共享同一个Address对象。

3. 深拷贝(Deep Copy)

深拷贝不仅创建一个新对象,还会递归地复制所有成员对象。这样,原对象和新对象完全独立,不共享任何引用。

深拷贝可以通过手动实现clone方法来完成,或者使用序列化。

手动实现深拷贝的示例:

class Address implements Cloneable {String city;public Address(String city) {this.city = city;}@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}class Person implements Cloneable {String name;Address address;public Person(String name, Address address) {this.name = name;this.address = address;}@Overrideprotected Object clone() throws CloneNotSupportedException {Person cloned = (Person) super.clone();cloned.address = (Address) address.clone();  // 深拷贝return cloned;}
}public class Main {public static void main(String[] args) throws CloneNotSupportedException {Address address = new Address("New York");Person person1 = new Person("Alice", address);Person person2 = (Person) person1.clone();person2.name = "Bob";person2.address.city = "Los Angeles";System.out.println(person1.name); // 输出 "Alice"System.out.println(person1.address.city); // 输出 "New York"System.out.println(person2.name); // 输出 "Bob"System.out.println(person2.address.city); // 输出 "Los Angeles"}
}

在上述代码中,person1person2的所有属性都是独立的,修改一个对象的属性不会影响另一个对象。
通过序列化和反序列化来实现Java的深拷贝是一种通用且方便的方法。它可以确保对象的完整复制,包括所有嵌套的成员对象。以下是具体的实现步骤:

  1. 让类实现Serializable接口:确保需要深拷贝的类和它包含的所有成员类都实现Serializable接口。
  2. 使用序列化和反序列化进行深拷贝:将对象写入字节流,然后从字节流中读出对象,从而实现对象的完全复制。

下面是一个具体的例子:

示例代码

import java.io.*;class Address implements Serializable {private static final long serialVersionUID = 1L;String city;public Address(String city) {this.city = city;}@Overridepublic String toString() {return "Address{city='" + city + "'}";}
}class Person implements Serializable {private static final long serialVersionUID = 1L;String name;Address address;public Person(String name, Address address) {this.name = name;this.address = address;}@Overridepublic String toString() {return "Person{name='" + name + "', address=" + address + "}";}// 深拷贝方法public Person deepCopy() {try {// 序列化ByteArrayOutputStream bos = new ByteArrayOutputStream();ObjectOutputStream oos = new ObjectOutputStream(bos);oos.writeObject(this);oos.flush();oos.close();// 反序列化ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());ObjectInputStream ois = new ObjectInputStream(bis);return (Person) ois.readObject();} catch (IOException | ClassNotFoundException e) {e.printStackTrace();return null;}}
}public class Main {public static void main(String[] args) {Address address = new Address("New York");Person person1 = new Person("Alice", address);Person person2 = person1.deepCopy();person2.name = "Bob";person2.address.city = "Los Angeles";System.out.println("Original: " + person1);  // 原对象System.out.println("Copy: " + person2);  // 深拷贝后的对象}
}

代码解释

  1. 实现Serializable接口AddressPerson类都实现了Serializable接口,使它们可以被序列化。
  2. 深拷贝方法deepCopy
    • 序列化:使用ObjectOutputStream将对象写入ByteArrayOutputStream
    • 反序列化:使用ObjectInputStreamByteArrayInputStream中读出对象,生成一个新的对象副本。
  3. 测试深拷贝
    • 创建原始对象person1,并通过deepCopy方法生成person2
    • 修改person2的属性,验证原始对象person1不受影响,证明了对象的深拷贝。

这种方法的优点是实现简单,且适用于所有需要深拷贝的情况。然而,它也有一些限制,比如性能较慢(因为涉及IO操作)和必须实现Serializable接口。

总结

  • 引用拷贝:只复制引用,原对象和新对象指向同一个对象。
  • 浅拷贝:创建新对象,但不递归复制成员对象,成员对象仍然是共享的引用。
  • 深拷贝:创建新对象,并递归复制所有成员对象,完全独立。

这些拷贝方式在实际应用中有不同的使用场景和适用性,根据需要选择合适的拷贝方式可以有效管理内存和对象关系。

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

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

相关文章

数字IC后端物理验证PV | TSMC 12nm Calibre Base Layer DRC案例解析

基于TSMC 12nm ARM A55 upf flow后端设计实现训练营将于6月中旬正式开班!小班教学!目前还有3个名额,招满为止!有需要可以私信小编 ic-backend2018报名。吾爱IC社区所有训练营课程均为直播课! 这个课程支持升级成双核A…

服务器禁止密码登陆

转载请标明出处:https://blog.csdn.net/donkor_/article/details/139444224 文章目录 一、前言二、编辑sshd_config文件三、重启服务四、总结 一、前言 复杂的密码,登陆服务器的时候,也是很不方便的。并且频繁登陆,暴露给外界&am…

事件总线vueEvent

一个组件结束后要更新另一个组件数据,但是另一个组件和这个组件没有上下级关系 在 Vue 中,非父子组件之间进行通信通常需要使用事件总线或者其他的全局事件管理器。在你的代码片段中,vueEvent 似乎是一个事件总线对象,通过 emit 方…

c++ 里函数选择的优先级:普通函数、模板函数、万能引用,编译器选择哪个执行呢?

看大师写的代码时,除了在类里定义了 copy 构造函数,移动构造函数,还定义了对形参采取万能引用的构造函数,因此有个疑问,这时候的构造函数优先级是什么样的呢?简化逻辑测试一下,如下图&#xff0…

如何实现JavaScript中的寄生组合式继承?

在JavaScript中,寄生组合式继承是一种继承机制,它结合了寄生式继承和组合继承的特点。其核心思想是通过构造函数来继承属性,同时通过原型链来继承方法。以下是实现寄生组合式继承的基本步骤: 首先定义一个辅助函数 inheritProtot…

Pygame:新手指南与入门教程

在游戏开发领域,pygame 是一个广受欢迎的 Python 库,它提供了开发二维游戏的丰富工具和方法。这个库让开发者可以较少地关注底层图形处理细节,更多地专注于游戏逻辑和玩法的实现。本文将详细介绍 pygame,包括其安装过程、基本概念、主要功能和一个简单游戏的开发流程。 一…

【Vue】路由的封装抽离

问题:所有的路由配置都在main.js中合适吗? 目标:将路由模块抽离出来。 好处:拆分模块,利于维护 路径简写: 脚手架环境下 指代src目录,可以用于快速引入组件 完整代码 router/index.js // 但…

探索贷款交易平台的技术架构与创新应用

随着金融科技的快速发展,贷款交易平台作为金融行业的重要组成部分,正扮演着越来越重要的角色。本文将深入探讨贷款交易平台的技术架构和创新应用,从前端设计、后端系统、安全保障和智能化服务等方面进行全面解析,帮助读者更好地了…

【Python报错】已解决AttributeError: list object has no attribute ’shape‘ ( Solved )

解决Python报错:AttributeError: ‘list’ object has no attribute ‘shape’ (Solved) 在Python中,AttributeError表明你试图访问的对象没有你请求的属性或方法。如果你遇到了AttributeError: list object has no attribute shape的错误,这…

为什么要用Git

1. Git是什么 1.1. 概述 Git是分布式版本控制系统,与SVN类似的集中化版本控制系统相比,集中化版本控制系统如果中央服务器宕机则会影响数据和协同开发。 Git是分布式的版本控制系统,客户端不只是提取最新版本的快照,而且将整个…

【Java毕业设计】基于Java的特色美食推荐网站的设计与实现

文章目录 摘 要ABSTRACT目 录1 概述1.1 研究背景及意义1.2 国内外研究现状1.3 拟研究内容1.4 系统开发技术1.4.1 Java编程语言1.4.2 SpringBoot框架1.4.3 MySQL数据库1.4.4 B/S结构1.4.5 MVC模式 2 系统需求分析2.1 可行性分析2.2 任务概述2.3 功能性需求3.2.2 数据库逻辑结构设…

全面解析如何租用免备案海外服务器

租用免备案海外服务器是许多企业和个人在全球范围内开展业务或访问国际互联网资源时选择的一种方式。这种服务具有无需经过中国互联网备案流程的优势,能够快速部署并使用。下面将详细介绍免备案海外服务器租用的相关信息,rak部落为您整理发布。 1. **国外…

外汇天眼:FSCS确认TenetConnect Services Ltd已任命管理人

2024年6月5日,Tenet Group Ltd的董事们任命了Interpath Ltd的Ed Boyle、Howard Smith和Rob Spence为联合管理人。Ed Boyle和Rob Spence也被任命为其子公司Tenet Ltd、TenetConnect Ltd和TenetConnect Services Ltd的联合管理人。Tenet Mortgage Services Ltd和Tenet…

【计算机视觉(8)】

基于Python的OpenCV基础入门——图像直方图 直方图图像直方图 图像直方图代码以及实现效果 直方图 直方图是一种用于描述图像亮度分布的统计工具。它将图像的像素亮度值按照不同的亮度等级进行计数,并以直方图的形式呈现出来。图像直方图可以显示图像中每个亮度级别…

点击式的excel电子表格查找修改功能,比xlookup和vlookup简单,多列关联查询速度更快

经过实际测试,excel的xlookup确实非常简单,有部分功能也非常快。但是有的人不会公式,或者不喜欢用公式,或者没有excel2021以上的版本。而且xlookup确实也有些还不是很完美的地方,比如对多列关联查询很慢。所以我们还是…

MVC前端怎么写:深入解析与实战指南

MVC前端怎么写:深入解析与实战指南 在Web开发领域,MVC(Model-View-Controller)是一种广泛使用的架构模式,它将应用程序的数据、界面和控制逻辑分离,使得代码更加清晰、易于维护。本文将详细探讨MVC前端如何…

selenium非全新的方式同时启动多个浏览器又互不影响的一种实现方法,欢迎讨论!

最近在做模拟浏览器批量定时自动点击实现批量操作功能,主要使用selenium,但是发现selenium直接调用本地浏览器,启动的是一个全新的(与手动打开的不一致),网站可以检测到,每次都要双重验证(密码登…

Windows系统中不同Java版本共存

Windows系统中不同Java版本共存的方法 在Windows系统中,有时我们需要同时运行多个Java应用,而这些应用可能依赖于不同版本的Java Development Kit (JDK) 或 Java Runtime Environment (JRE)。为了实现这种需求,我们需要在Windows中配置多个J…

我应该如何使用 Python 的 NLTK 库进行词频统计?

使用Python的NLTK(Natural Language Toolkit)库进行词频统计,你可以遵循以下步骤: 安装NLTK库: 如果你还没有安装NLTK,可以通过pip安装: pip install nltk导入必要的模块: 在Python脚…

电商APP用户体验提升技巧:一个实战案例

随着网络和移动技术的快速发展,加上全球疫情的影响,电子商务应用程序改变了人们的购物方式,积累了大量的用户群体。如今,一个成功的电子商务应用程序,除了网站用户界面的美,电子商务用户体验的设计&#xf…