Java深度复制功能与用法实例分析
作者:PursueCloud 发布时间:2023-07-05 11:23:17
标签:Java,深度复制
本文实例讲述了Java深度复制功能与用法。分享给大家供大家参考,具体如下:
写在前面:
什么是深度复制?在Java里面,在创建一个对象,我们通常会有一个引用指向该对象,当我们通过引用变量改变对象的值(属性)时,引用是不变的,变的是内存里面的那块内存,即引用所指向的对象。一般情况下,我们将该引用赋给另一个引用变量或者作为参数传递时,传递的也只是引用,即将引用指向“复制”了一份给另一个引用变量,随后该引用变量也指向同一个对象,内存里面并没有创建一个新的对象。在某些情况下,我们需要“真正复制”对象,创建一份已知对象的copy,而不仅仅“复制”引用,用作备份也好,其他操作也好。
那么,该如何实现?
先说下思路:首先将对象序列化到流里,然后再反序列化,从流里读取出来即可。
下面上代码:
package com.yo.java;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;
/**
* Java 实现深度复制
* @author Yo
*
*/
public class DeepCopy implements Serializable{
int i;
/**
* @param args
* @throws IOException
* @throws ClassNotFoundException
*/
public static void main(String[] args) throws ClassNotFoundException, IOException {
demo1();
demo2();
}
/**
* 深度复制,实参类必须实现Serializable接口
* @param o
* @return
* @throws IOException
* @throws ClassNotFoundException
*/
public static Object deepCopy(Object o) throws IOException, ClassNotFoundException {
// //先序列化,写入到流里
ByteArrayOutputStream bo = new ByteArrayOutputStream();
ObjectOutputStream oo = new ObjectOutputStream(bo);
oo.writeObject(o);
//然后反序列化,从流里读取出来,即完成复制
ByteArrayInputStream bi = new ByteArrayInputStream(bo.toByteArray());
ObjectInputStream oi = new ObjectInputStream(bi);
return oi.readObject();
}
/**
* 引用传递与深度复制
* @throws ClassNotFoundException
* @throws IOException
*/
public static void demo1() throws ClassNotFoundException, IOException {
System.out.println("===========未使用深度复制=========");
DeepCopy dc1 = new DeepCopy();
dc1.i = 1;//初始化dc1里i的值
DeepCopy dc2 = dc1;
dc1.i = 2;//改变dc1里i的值
System.out.println("dc1 : " + dc1.i);
System.out.println("dc2(引用传递) : " + dc2.i);
System.out.println("===========使用深度复制=========");
DeepCopy dc3 = new DeepCopy();
dc3.i = 1;//初始化dc3里i的值
DeepCopy dc4 = (DeepCopy)deepCopy(dc3);
dc3.i = 2;//改变dc3里i的值
System.out.println("dc3 : " + dc3.i);
System.out.println("dc4(深度复制) : " + dc4.i);
}
/**
* 集合的值复制与深度复制
* @throws ClassNotFoundException
* @throws IOException
*/
public static void demo2() throws ClassNotFoundException, IOException {
System.out.println("===========未使用深度复制=========");
//创建DeepCopy对象,并初始化i的值,添加到list1集合
DeepCopy dc = new DeepCopy();
dc.i = 1;//初始化dc1里i的值
List<DeepCopy> list1 = new ArrayList<DeepCopy>();
list1.add(dc);
//未使用深度复制
List<DeepCopy> list2 = new ArrayList<DeepCopy>(list1);//这里与使用Collections.copy(dest, src)结果一样
//改变list1中元素的值
for(DeepCopy d1 : list1) {
//改变dc1里i的值
d1.i = 2;
}
//遍历list
for(DeepCopy d1 : list1) {
System.out.println("list1 : " + d1.i);
}
//遍历list2
for(DeepCopy d2 : list2) {
System.out.println("list2(复制) : " + d2.i);
}
System.out.println("===========使用深度复制后=========");
DeepCopy dc3 = new DeepCopy();
dc3.i = 1;//初始化dc3里i的值
List<DeepCopy> list3 = new ArrayList<DeepCopy>();
list3.add(dc3);
List<DeepCopy> list4 = (List<DeepCopy>) deepCopy(list3);
for(DeepCopy d : list3) {
//改变dc3里i的值
d.i = 2;
}
for(DeepCopy d3 : list3) {
System.out.println("list3 : " + d3.i);
}
for(DeepCopy d4 : list4) {
System.out.println("list4(深度复制): " + d4.i);
}
}
}
以上运行结果如下(实测):
===========未使用深度复制=========
dc1 : 2
dc2(引用传递) : 2
===========使用深度复制=========
dc3 : 2
dc4(深度复制) : 1
===========未使用深度复制=========
list1 : 2
list2(复制) : 2
===========使用深度复制后=========
list3 : 2
list4(深度复制): 1
由此可见,当仅仅只是引用传递或者根据对象的值创建新的值,仅能称为“浅复制”,当原对象的属性发生改变时,根据上面方式创建的新对象的属性也会随之改变;而如果采用深度复制,那是真正的copy了一份新的对象,新对象的与原对象不存在任何关联,原对象的属性发生改变不会影响新对象,就如同copy的意义一样
如上,如有不妥,如能指出,非常感谢
更多java相关内容感兴趣的读者可查看本站专题:《Java数据结构与算法教程》、《Java操作DOM节点技巧总结》、《Java文件与目录操作技巧汇总》和《Java缓存操作技巧汇总》
希望本文所述对大家java程序设计有所帮助。
来源:http://blog.csdn.net/u011437847/article/details/52261191
![](https://www.aspxhome.com/images/zang.png)
![](https://www.aspxhome.com/images/jiucuo.png)
猜你喜欢
- 实现Java多态性的时候,关于方法调用的优先级:我们这样假设下,super(超类)、this(当前类对象)、show(方法)、object(
- 提示:这里咱们要说的常量池,常量池就是咱们面试中所说的常量池,谈谈你对常量池的认识?面试官一问咱们就懵逼了,你要记得你脑子中有一张图!!!
- 众所周知Web服务器与客户端之间的通信是使用HTTP协议的。HTTP是一个客户端和服务器端请求和应答的标准(TCP)。因为HTTP协议是基于
- 现在Java的大部分项目都是基于Maven, 在Maven项目中使用Selenium2. 非常简单。 首先你需要配置好
- Linux下的五种I/O模型1)阻塞I/O(blocking I/O)2)非阻塞I/O (nonblocking I/O)3) I/O复用(
- 这篇文章写的非常好,深入浅出,关键还是一位大三学生自己剖析的心得。这是我喜欢此文的原因。下面请看正文:作为一个大三的预备程序员,我学习and
- 可变数目参数的好处就是在某些情况下可以方便地对参数个数不确定情况的实现,例如计算任意数字的加权和,链接任意字符串为一个字符串等。看下例子:p
- .NETCORE 中的 Generic Host本文以自己在工作中学习和使用.net core generic-host 作一个总结。前言在
- mybatis oracle 自增序列的代码如下所示:<insert id=" insert " useGener
- 今天朋友圈又火了,听说原因是 @腾讯官网 就能得到一顶绿色的帽子,啊呸,是一个好看的国庆节头像,可是听说没一会就502了,那么我们自己动手实
- @Aspect中有5种通知@Before:前置通知, 在方法执行之前执行@Aroud:环绕通知, 围绕着方法执行@After:后置通知, 在
- 最近由于项目需求,项目中需要实现一个WebSSH连接终端的功能,由于自己第一次做这类型功能,所以首先上了GitHub找了找有没有现成的轮子可
- NPOI是一个开源的C#读写Excel、WORD等微软OLE2组件文档的项目。使用 NPOI 可以在没有安装 Office 或者相应环境的机
- 写在前面jenkins作为java的好 * ,经历过单体项目时代->集群项目时代->容器集群分布式时代,使用稳定可靠,cpu友好(
- 定义:定义一个用于创建对象的接口,让子类决定实例化哪一个类,工厂方法使一个类的实例化延迟到其子类。类型:创建类模式类图:工厂方法模式代码in
- 前段时间由于VSS上的一个项目要给2个公司开发使用,而2个公司的需求不同 就把该项目复制到VSS上的另外一个目录,结果在别人的电脑取出并打开
- 一、前言学习概述:学习四种不同类型的方法应用、方法被调用时的内存图、重载学习目标:熟练掌握方法的应用以及重载二、定义与调用1.概述定义:方法
- Solr我还是个菜鸟,写这一些文章只是记录一下最近一段时间学习Solr的心得。 Solr是什么? 最近我学Solr的时候,一直看到一句话,S
- 井字棋游戏要求在3乘3棋盘上,每行都相同或者每列都相同再或者对角线相同,则胜出.因此我们可以使用一个二维数组来表示棋盘,判断胜负只需要判断数
- 第1部分 HashSet介绍HashSet 简介HashSet 是一个没有重复元素的集合。它是由HashMap实现的,不保证元素的顺序,而且