Java中的对象拷贝(Object Copy)指的是将一个对象的所有属性(成员变量)拷贝到另一个有着相同类类型的对象中去。举例说明:比如,对象A和对象B都属于类S,具有属性a和b。那么对对象A进行拷贝操作赋值给对象B就是:B.a=A.a;  B.b=A.b;

在程序中拷贝对象是很常见的,主要是为了在新的上下文环境中复用现有对象的部分或全部 数据。

Java中的对象拷贝主要分为:浅拷贝(Shallow Copy)、深拷贝(Deep Copy)。

先介绍一点铺垫知识:Java中的数据类型分为基本数据类型和引用数据类型。对于这两种数据类型,在进行赋值操作、用作方法参数或返回值时,会有值传递和引用(地址)传递的差别。

浅拷贝(Shallow Copy):①对于数据类型是基本数据类型的成员变量,浅拷贝会直接进行值传递,也就是将该属性值复制一份给新的对象。因为是两份不同的数据,所以对其中一个对象的该成员变量值进行修改,不会影响另一个对象拷贝得到的数据。②对于数据类型是引用数据类型的成员变量,比如说成员变量是某个数组、某个类的对象等,那么浅拷贝会进行引用传递,也就是只是将该成员变量的引用值(内存地址)复制一份给新的对象。因为实际上两个对象的该成员变量都指向同一个实例。在这种情况下,在一个对象中修改该成员变量会影响到另一个对象的该成员变量值。

具体模型如图所示:可以看到基本数据类型的成员变量,对其值创建了新的拷贝。而引用数据类型的成员变量的实例仍然是只有一份,两个对象的该成员变量都指向同一个实例。

浅拷贝的实现方式主要有三种:

一、通过拷贝构造方法实现浅拷贝:

拷贝构造方法指的是该类的构造方法参数为该类的对象。使用拷贝构造方法可以很好地完成浅拷贝,直接通过一个现有的对象创建出与该对象属性相同的新的对象。

代码参考如下:

/* 拷贝构造方法实现浅拷贝 */
public class CopyConstructor {public static void main(String[] args) {Age a=new Age(20);Person p1=new Person(a,"摇头耶稣");Person p2=new Person(p1);System.out.println("p1是"+p1);System.out.println("p2是"+p2);//修改p1的各属性值,观察p2的各属性值是否跟随变化p1.setName("小傻瓜");a.setAge(99);System.out.println("修改后的p1是"+p1);System.out.println("修改后的p2是"+p2);}
}class Person{//两个属性值:分别代表值传递和引用传递private Age age;private String name;public Person(Age age,String name) {this.age=age;this.name=name;}//拷贝构造方法public Person(Person p) {this.name=p.name;this.age=p.age;}public void setName(String name) {this.name=name;}public String toString() {return this.name+" "+this.age;}
}class Age{private int age;public Age(int age) {this.age=age;}public void setAge(int age) {this.age=age;}public int getAge() {return this.age;}public String toString() {return getAge()+"";}
}

运行结果为:

p1是摇头耶稣 20
p2是摇头耶稣 20
修改后的p1是小傻瓜 99
修改后的p2是摇头耶稣 99

结果分析:这里对Person类选择了两个具有代表性的属性值:一个是引用传递类型;另一个是字符串类型(属于常量)。

通过拷贝构造方法进行了浅拷贝,各属性值成功复制。其中,p1值传递部分的属性值发生变化时,p2不会随之改变;而引用传递部分属性值发生变化时,p2也随之改变。

要注意:如果在拷贝构造方法中,对引用数据类型变量逐一开辟新的内存空间,创建新的对象,也可以实现深拷贝。而对于一般的拷贝构造,则一定是浅拷贝。

二、通过重写clone()方法进行浅拷贝:

Object类是类结构的根类,其中有一个方法为protected Object clone() throws CloneNotSupportedException,这个方法就是进行的浅拷贝。有了这个浅拷贝模板,我们可以通过调用clone()方法来实现对象的浅拷贝。但是需要注意:1、Object类虽然有这个方法,但是这个方法是受保护的(被protected修饰),所以我们无法直接使用。2、使用clone方法的类必须实现Cloneable接口,否则会抛出异常CloneNotSupportedException。对于这两点,我们的解决方法是,在要使用clone方法的类中重写clone()方法,通过super.clone()调用Object类中的原clone方法。

参考代码如下:对Student类的对象进行拷贝,直接重写clone()方法,通过调用clone方法即可完成浅拷贝。

/* clone方法实现浅拷贝 */
public class ShallowCopy {public static void main(String[] args) {Age a=new Age(20);Student stu1=new Student("摇头耶稣",a,175);//通过调用重写后的clone方法进行浅拷贝Student stu2=(Student)stu1.clone();System.out.println(stu1.toString());System.out.println(stu2.toString());//尝试修改stu1中的各属性,观察stu2的属性有没有变化stu1.setName("大傻子");//改变age这个引用类型的成员变量的值a.setAge(99);//stu1.setaAge(new Age(99));    使用这种方式修改age属性值的话,stu2是不会跟着改变的。因为创建了一个新的Age类对象而不是改变原对象的实例值stu1.setLength(216);System.out.println(stu1.toString());System.out.println(stu2.toString());}
}/** 创建年龄类*/
class Age{//年龄类的成员变量(属性)private int age;//构造方法public Age(int age) {this.age=age;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public String toString() {return this.age+"";}
}
/** 创建学生类*/
class Student implements Cloneable{//学生类的成员变量(属性),其中一个属性为类的对象private String name;private Age aage;private int length;//构造方法,其中一个参数为另一个类的对象public Student(String name,Age a,int length) {this.name=name;this.aage=a;this.length=length;}//eclipe中alt+shift+s自动添加所有的set和get方法public String getName() {return name;}public void setName(String name) {this.name = name;}public Age getaAge() {return this.aage;}public void setaAge(Age age) {this.aage=age;}public int getLength() {return this.length;}public void setLength(int length) {this.length=length;}//设置输出的字符串形式public String toString() {return "姓名是: "+this.getName()+", 年龄为: "+this.getaAge().toString()+", 长度是: "+this.getLength();}//重写Object类的clone方法public Object clone() {Object obj=null;//调用Object类的clone方法,返回一个Object实例try {obj= super.clone();} catch (CloneNotSupportedException e) {e.printStackTrace();}return obj;}
}

运行结果如下:

姓名是: 摇头耶稣, 年龄为: 20, 长度是: 175
姓名是: 摇头耶稣, 年龄为: 20, 长度是: 175
姓名是: 大傻子, 年龄为: 99, 长度是: 216
姓名是: 摇头耶稣, 年龄为: 99, 长度是: 175

其中:Student类的成员变量我有代表性地设置了三种:基本数据类型的成员变量length,引用数据类型的成员变量aage和字符串String类型的name.

分析结果可以验证:

基本数据类型是值传递,所以修改值后不会影响另一个对象的该属性值;

引用数据类型是地址传递(引用传递),所以修改值后另一个对象的该属性值会同步被修改。

String类型非常特殊,所以我额外设置了一个字符串类型的成员变量来进行说明。首先,String类型属于引用数据类型,不属于基本数据类型,但是String类型的数据是存放在常量池中的,也就是无法修改的!也就是说,当我将name属性从“摇头耶稣”改为“大傻子"后,并不是修改了这个数据的值,而是把这个数据的引用从指向”摇头耶稣“这个常量改为了指向”大傻子“这个常量。在这种情况下,另一个对象的name属性值仍然指向”摇头耶稣“不会受到影响。

深拷贝:首先介绍对象图的概念。设想一下,一个类有一个对象,其成员变量中又有一个对象,该对象指向另一个对象,另一个对象又指向另一个对象,直到一个确定的实例。这就形成了对象图。那么,对于深拷贝来说,不仅要复制对象的所有基本数据类型的成员变量值,还要为所有引用数据类型的成员变量申请存储空间,并复制每个引用数据类型成员变量所引用的对象,直到该对象可达的所有对象。也就是说,对象进行深拷贝要对整个对象图进行拷贝!

简单地说,深拷贝对引用数据类型的成员变量的对象图中所有的对象都开辟了内存空间;而浅拷贝只是传递地址指向,新的对象并没有对引用数据类型创建内存空间。

深拷贝模型如图所示:可以看到所有的成员变量都进行了复制。

因为创建内存空间和拷贝整个对象图,所以深拷贝相比于浅拷贝速度较慢并且花销较大。

深拷贝的实现方法主要有两种:

一、通过重写clone方法来实现深拷贝

与通过重写clone方法实现浅拷贝的基本思路一样,只需要为对象图的每一层的每一个对象都实现Cloneable接口并重写clone方法,最后在最顶层的类的重写的clone方法中调用所有的clone方法即可实现深拷贝。简单的说就是:每一层的每个对象都进行浅拷贝=深拷贝。

参考代码如下:

package linearList;
/* 层次调用clone方法实现深拷贝 */
public class DeepCopy {public static void main(String[] args) {Age a=new Age(20);Student stu1=new Student("摇头耶稣",a,175);//通过调用重写后的clone方法进行浅拷贝Student stu2=(Student)stu1.clone();System.out.println(stu1.toString());System.out.println(stu2.toString());System.out.println();//尝试修改stu1中的各属性,观察stu2的属性有没有变化stu1.setName("大傻子");//改变age这个引用类型的成员变量的值a.setAge(99);//stu1.setaAge(new Age(99));    使用这种方式修改age属性值的话,stu2是不会跟着改变的。因为创建了一个新的Age类对象而不是改变原对象的实例值stu1.setLength(216);System.out.println(stu1.toString());System.out.println(stu2.toString());}
}/** 创建年龄类*/
class Age implements Cloneable{//年龄类的成员变量(属性)private int age;//构造方法public Age(int age) {this.age=age;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public String toString() {return this.age+"";}//重写Object的clone方法public Object clone() {Object obj=null;try {obj=super.clone();} catch (CloneNotSupportedException e) {e.printStackTrace();}return obj;}
}
/** 创建学生类*/
class Student implements Cloneable{//学生类的成员变量(属性),其中一个属性为类的对象private String name;private Age aage;private int length;//构造方法,其中一个参数为另一个类的对象public Student(String name,Age a,int length) {this.name=name;this.aage=a;this.length=length;}public String getName() {return name;}public void setName(String name) {this.name = name;}public Age getaAge() {return this.aage;}public void setaAge(Age age) {this.aage=age;}public int getLength() {return this.length;}public void setLength(int length) {this.length=length;}public String toString() {return "姓名是: "+this.getName()+", 年龄为: "+this.getaAge().toString()+", 长度是: "+this.getLength();}//重写Object类的clone方法public Object clone() {Object obj=null;//调用Object类的clone方法——浅拷贝try {obj= super.clone();} catch (CloneNotSupportedException e) {e.printStackTrace();}//调用Age类的clone方法进行深拷贝//先将obj转化为学生类实例Student stu=(Student)obj;//学生类实例的Age对象属性,调用其clone方法进行拷贝stu.aage=(Age)stu.getaAge().clone();return obj;}
}

姓名是: 摇头耶稣, 年龄为: 20, 长度是: 175
姓名是: 摇头耶稣, 年龄为: 20, 长度是: 175
姓名是: 大傻子, 年龄为: 99, 长度是: 216
姓名是: 摇头耶稣, 年龄为: 20, 长度是: 175

分析结果可以验证:进行了深拷贝之后,无论是什么类型的属性值的修改,都不会影响另一个对象的属性值。

二、通过对象序列化实现深拷贝

虽然层次调用clone方法可以实现深拷贝,但是显然代码量实在太大。特别对于属性数量比较多、层次比较深的类而言,每个类都要重写clone方法太过繁琐。

将对象序列化为字节序列后,默认会将该对象的整个对象图进行序列化,再通过反序列即可完美地实现深拷贝。

参考代码如下:

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;/* 通过序列化实现深拷贝 */
public class DeepCopyBySerialization {public static void main(String[] args) throws IOException, ClassNotFoundException  {Age a=new Age(20);Student stu1=new Student("摇头耶稣",a,175);//通过序列化方法实现深拷贝ByteArrayOutputStream bos=new ByteArrayOutputStream();ObjectOutputStream oos=new ObjectOutputStream(bos);oos.writeObject(stu1);oos.flush();ObjectInputStream ois=new ObjectInputStream(new ByteArrayInputStream(bos.toByteArray()));Student stu2=(Student)ois.readObject();System.out.println(stu1.toString());System.out.println(stu2.toString());System.out.println();//尝试修改stu1中的各属性,观察stu2的属性有没有变化stu1.setName("大傻子");//改变age这个引用类型的成员变量的值a.setAge(99);stu1.setLength(216);System.out.println(stu1.toString());System.out.println(stu2.toString());}
}/** 创建年龄类*/
class Age implements Serializable{//年龄类的成员变量(属性)private int age;//构造方法public Age(int age) {this.age=age;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public String toString() {return this.age+"";}
}
/** 创建学生类*/
class Student implements Serializable{//学生类的成员变量(属性),其中一个属性为类的对象private String name;private Age aage;private int length;//构造方法,其中一个参数为另一个类的对象public Student(String name,Age a,int length) {this.name=name;this.aage=a;this.length=length;}//eclipe中alt+shift+s自动添加所有的set和get方法public String getName() {return name;}public void setName(String name) {this.name = name;}public Age getaAge() {return this.aage;}public void setaAge(Age age) {this.aage=age;}public int getLength() {return this.length;}public void setLength(int length) {this.length=length;}//设置输出的字符串形式public String toString() {return "姓名是: "+this.getName()+", 年龄为: "+this.getaAge().toString()+", 长度是: "+this.getLength();}
}

运行结果为:

姓名是: 摇头耶稣, 年龄为: 20, 长度是: 175
姓名是: 摇头耶稣, 年龄为: 20, 长度是: 175
姓名是: 大傻子, 年龄为: 99, 长度是: 216
姓名是: 摇头耶稣, 年龄为: 20, 长度是: 175

可以通过很简洁的代码即可完美实现深拷贝。不过要注意的是,如果某个属性被transient修饰,那么该属性就无法被拷贝了。

三、json序列化方式

4. json序列化

public class JsonCopy {private static ObjectMapper mapper = new ObjectMapper(); public static String encodeWithoutNull(Object obj) throws Exception { return mapper.writeValueAsString(obj); } public static <T> T decodeValueIgnoreUnknown(String str, Class<T> clazz) throws Exception { return mapper.readValue(str, clazz); } // 一千万次 15.3秒 public static <T> T copy(T source, Class<T> tClass) throws Exception { return decodeValueIgnoreUnknown(encodeWithoutNull(source), tClass); } }

以上是浅拷贝的深拷贝的区别和实现方式。

### 浅拷贝的补充

Spring的beanutils的copypropertires是浅拷贝的实现方式。

文章转载自:https://www.cnblogs.com/shakinghead/p/7651502.html

Java的浅拷贝与深拷贝总结相关推荐

  1. java中浅拷贝和深拷贝_java中的浅拷贝和深拷贝

    复制 将一个对象的引用复制给另一个对象,一共有三种方式.第一种方式是直接赋值,第二种方式是浅复制,第三种方式是深复制. 1.直接赋值 在Java中,A a1 = a2,这实际上复制的是引用,也就是说 ...

  2. Java中浅拷贝与深拷贝之间的区别

    在深入探讨Java中浅表副本与深表副本之间的差异之前,让我们看看首先进行克隆的是什么. 什么是克隆? 克隆是在内存中创建现有对象的精确副本的过程.在Java中,java.lang.Object类的cl ...

  3. Java中浅拷贝和深拷贝的区别

    深拷贝和浅拷贝的区别 浅拷贝:被拷贝的对象的所有属性值都与原来的对象相同,而对象的所有属性引用仍然指向原来的属性所指向的内存地址.需要注意的是cloneObj == obj 返回的是false,所以使 ...

  4. 基于java实现浅拷贝和深拷贝

    目录 1.概念 2.浅拷贝 2.1.浅拷贝实战 3.深拷贝 3.1.嵌套 clone 方法 3.2.使用序列化流 3.3.使用开源工具类 1.概念 浅拷贝:在拷贝一个对象时,复制基本数据类型的成员变量 ...

  5. Java【浅拷贝和深拷贝】之间的区别

    浅拷贝 浅拷贝定义定义 被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象.即对象的浅拷贝会对"主"对象进行拷贝,但不会复制主对象里面所引用 ...

  6. Java基础—复制之深拷贝与浅拷贝

    目录 一.浅拷贝(Shallow Copy) 二.深拷贝(Deep Copy) Java中的对象拷贝(Object Copy)指的是将一个对象的所有属性(成员变量)拷贝到另一个有着相同类类型的对象中去 ...

  7. java深入理解浅拷贝和深拷贝

    文章目录 简介 拷贝接口 使用clone导致的浅拷贝 使用clone的深拷贝 不要overridden clone 总结 简介 拷贝对象是java中经常会遇到的问题.java中存在两种类型,基础类型和 ...

  8. Java提高篇 —— Java浅拷贝和深拷贝

    一.前言 我们知道在Java中存在这个接口Cloneable,实现该接口的类都会具备被拷贝的能力,同时拷贝是在内存中进行,在性能方面比我们直接通过new生成对象来的快,特别是在大对象的生成上,使得性能 ...

  9. java数组深拷贝和浅拷贝_java List复制:浅拷贝与深拷贝

    Java的拷贝可以分为三种:浅拷贝(Shallow Copy).深拷贝(Deep Copy).延迟拷贝(Lazy Copy). 在java中除了基本数据类型之外(int,long,short等),还存 ...

最新文章

  1. 此时不应有java_Java 锁的知识总结及实例代码
  2. [投稿]通过Web界面在多台服务器上批量创建文件
  3. VMWare虚拟机连接方式
  4. Laravel Auth 自定义user 模型目录结构
  5. 超实用!VLAN、TRUNK、VLAN间路由基础
  6. [转贴]犯贱报(一张浓缩大学生活的..)
  7. swagger传递日期类型 json形式
  8. 作者:聂敏,男,电子科技大学教育大数据研究所博士生。
  9. 计算机专业大学排名_最新!2020美国九大热门专业最具薪资潜力大学排名来了!...
  10. Windows域控设置IE主页 默认打开百度 【全域策略生效】
  11. pytorch学习笔记(三十一):门控循环单元(GRU)
  12. Webpack入门——使用Webpack打包Angular项目的一个例子
  13. ABP应用层——参数有效性验证
  14. 20190818 On Java8 第八章 复用
  15. AppTheme属性设置集合
  16. 08. 切勿创建包含auto_ptr的容器
  17. 【HUST】网安|编译原理实验|实验四攻略
  18. knockout的监控数组实现 - 司徒正美
  19. 分享一个强大的数据可视化低代码开发平台
  20. BUUCTF_Crypto_[MRCTF2020]天干地支+甲子

热门文章

  1. matlab设计pss参考信号,[OAI][Layer1]PSS/SSS procedure
  2. 3人2周上线,2人1周上线,Solo明天上线!开发周期果然不能用搬砖模式计算......
  3. 每日一皮:终于明白女朋友挂在嘴边的“鸽子蛋”为什么那么贵了...
  4. 我说分布式事务之TCC
  5. matlab微分方程组边值,matlab求解常微分方程边值问题的方法
  6. 苏州大学9月计算机考试试题,2016年9月计算机一级考试题及答案
  7. pytorch 内积 矩阵乘法
  8. qgridlayout 动态刷新
  9. python 时间字符串
  10. PyCharm中控制台输出日志分层级分颜色显示