Java Object 对象拷贝

@author ixenos

JAVA 对象拷贝


Java里的clone分为:   

  1.浅拷贝:浅复制仅仅复制所考虑的对象,而不复制它所引用的对象,Object类里的clone方法是浅拷贝   

  2.深拷贝:深复制把要复制的对象及其所引用的对象都复制了一遍

  3.序列化深拷贝:把对象写到流里的过程是序列化(Serilization)过程,而把对象从流中读出来的过程则叫做反序列化(Deserialization)过程。应当指出的是,写在流里的是对象的一个拷贝,而原对象仍然存在于JVM里面,利用这个特性,可以做深拷贝

深拷贝时必须要遵循下面三点:
  1.在派生类中覆盖基类的clone()方法,并声明为public(Object类中的clone()方法为protected的)
  2.在派生类的clone()方法中,调用super.clone()
  3.在派生类中实现Cloneable接口

示例


浅拷贝的例子:

public class CloneTest
{ public static void main(String[] args) throws Exception
{
// teacher对象将被clone出来的Student对象共享.
Teacher teacher = new Teacher();
teacher.setAge(40);
teacher.setName("Teacher zhang"); Student student1 = new Student();
student1.setAge(20);
student1.setName("zhangsan");
student1.setTeacher(teacher); // 复制出来一个对象student2
Student student2 = (Student) student1.clone();
System.out.println(student2.getAge());
System.out.println(student2.getName()); System.out.println("~~~~~~~~~~~~~~~~~~~~~~");
System.out.println(student1.getTeacher().getAge());
System.out.println(student1.getTeacher().getName()); // 修改student2的引用对象
student2.getTeacher().setAge(50);
student2.getTeacher().setName("Teacher Li"); System.out.println("~~~~~~~~~~~~~~~~~~~~~~");
System.out.println(student1.getTeacher().getAge());
System.out.println(student1.getTeacher().getName());
}
} class Teacher
{
public int age;
public String name; public int getAge()
{
return age;
} public void setAge(int age)
{
this.age = age;
} public String getName()
{
return name;
} public void setName(String name)
{
this.name = name;
} } class Student implements Cloneable
{ public int age;
public String name;
public Teacher teacher; public int getAge()
{
return age;
} public void setAge(int age)
{
this.age = age;
} public String getName()
{
return name;
} public void setName(String name)
{
this.name = name;
} public Teacher getTeacher()
{
return teacher;
} public void setTeacher(Teacher teacher)
{
this.teacher = teacher;
} @Override
public Object clone() throws CloneNotSupportedException
{
return super.clone();
}
}
输出结果为:
20
zhangsan
~~~~~~~~~~~~~~~~~~~~~~
40
Teacher zhang
~~~~~~~~~~~~~~~~~~~~~~
50
Teacher Li

深拷贝例子:

public class DeepCloneTest
{ public static void main(String[] args) throws Exception
{
// teacher对象将不被clone出来的Student对象共享.
Teacher teacher = new Teacher();
teacher.setAge(40);
teacher.setName("Teacher zhang"); Student student1 = new Student();
student1.setAge(20);
student1.setName("zhangsan");
student1.setTeacher(teacher); // 复制出来一个对象student2
Student student2 = (Student) student1.clone();
System.out.println(student2.getAge());
System.out.println(student2.getName()); System.out.println("~~~~~~~~~~~~~~~~~~~~~~");
System.out.println(student1.getTeacher().getAge());
System.out.println(student1.getTeacher().getName()); // 修改student2的引用对象
student2.getTeacher().setAge(50);
student2.getTeacher().setName("Teacher Li"); System.out.println("~~~~~~~~~~~~~~~~~~~~~~");
System.out.println(student1.getTeacher().getAge());
System.out.println(student1.getTeacher().getName());
}
} class Teacher implements Cloneable
{
public int age;
public String name; public int getAge()
{
return age;
} public void setAge(int age)
{
this.age = age;
} public String getName()
{
return name;
} public void setName(String name)
{
this.name = name;
} @Override
public Object clone() throws CloneNotSupportedException
{
return super.clone();
} } class Student implements Cloneable
{ public int age;
public String name;
public Teacher teacher; public int getAge()
{
return age;
} public void setAge(int age)
{
this.age = age;
} public String getName()
{
return name;
} public void setName(String name)
{
this.name = name;
} public Teacher getTeacher()
{
return teacher;
} public void setTeacher(Teacher teacher)
{
this.teacher = teacher;
} @Override
public Object clone() throws CloneNotSupportedException
{
Student student = (Student) super.clone();
// 将引用的对象teacher也clone下
student.setTeacher((Teacher) (student.getTeacher().clone())); // student.getTeacher().clone()
return student;
}
} 输出结果为:
20
zhangsan
~~~~~~~~~~~~~~~~~~~~~~
40
Teacher zhang
~~~~~~~~~~~~~~~~~~~~~~
40
Teacher zhang

序列化深拷贝例子:

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
//利用序列化来做深复制
//深clone public class DeepCloneTest
{ public static void main(String[] args) throws Exception
{
// teacher对象将不被clone出来的Student对象共享.
Teacher teacher = new Teacher();
teacher.setAge(40);
teacher.setName("Teacher zhang"); Student student1 = new Student();
student1.setAge(20);
student1.setName("zhangsan");
student1.setTeacher(teacher); // 复制出来一个对象student2
Student student2 = (Student) student1.deepCopy();
System.out.println(student2.getAge());
System.out.println(student2.getName()); System.out.println("~~~~~~~~~~~~~~~~~~~~~~");
System.out.println(student1.getTeacher().getAge());
System.out.println(student1.getTeacher().getName()); // 修改student2的引用对象
student2.getTeacher().setAge(50);
student2.getTeacher().setName("Teacher Li"); System.out.println("~~~~~~~~~~~~~~~~~~~~~~");
System.out.println(student1.getTeacher().getAge());
System.out.println(student1.getTeacher().getName());
}
} class Teacher implements Serializable
{ private static final long serialVersionUID = -8834559347461591191L; public int age;
public String name; public int getAge()
{
return age;
} public void setAge(int age)
{
this.age = age;
} public String getName()
{
return name;
} public void setName(String name)
{
this.name = name;
} } class Student implements Serializable
{ // serialVersionUID
// 如果你的对象序列化后存到硬盘上面后,可是后来你却更改了类的field(增加或减少或改名),当你反序列化时,就会出现Exception的,这样就会造成不兼容性的问题。
// 但当serialVersionUID相同时,它就会将不一样的field以type的缺省值赋值(如int型的是0,String型的是null等),这个可以避开不兼容性的问题。所以最好给serialVersionUID赋值
private static final long serialVersionUID = 7991552226614088458L; public int age;
public String name;
public Teacher teacher;
public int getAge()
{
return age;
}
public void setAge(int age)
{
this.age = age;
}
public String getName()
{
return name;
}
public void setName(String name)
{
this.name = name;
} public Teacher getTeacher()
{
return teacher;
} public void setTeacher(Teacher teacher)
{
this.teacher = teacher;
}
/**
* 因为对象序列化会自然地对对象深拷贝,再利用“序列化-反序列化”读出
*/
public Object deepCopy() throws Exception
{
// 将该对象序列化成流,因为写在流里的是对象的一个拷贝,而原对象仍然存在于JVM里面。所以利用这个特性可以实现对象的深拷贝
ByteArrayOutputStream bos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(bos);
//this很关键,引用当前对象,当然,这是值传递
oos.writeObject(this); // 将流序列化成对象
ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray()); ObjectInputStream ois = new ObjectInputStream(bis); return ois.readObject();
}
} 输出结果为:
20
zhangsan
~~~~~~~~~~~~~~~~~~~~~~
40
Teacher zhang
~~~~~~~~~~~~~~~~~~~~~~
40
Teacher zhang
 

Java Object 对象拷贝的更多相关文章

  1. Java Object 对象拷贝答疑

    Java Object 对象拷贝答疑 @author ixenos 摘要:在对象的clone过程需要注意的几点.关于关键字this.super 关于clone[对象拷贝] 在实际编程过程,有时候我们会 ...

  2. Java Object 对象创建的方式 [ 转载 ]

    Java Object 对象创建的方式 [ 转载 ] @author http://blog.csdn.net/mhmyqn/article/details/7943411 显式创建 有4种显式地创建 ...

  3. Java中对象拷贝的两种方式

    引用的拷贝 //引用拷贝 private static void copyReferenceObject(){ Person p = new Person(23, "zhang") ...

  4. 通过与C++程序对比,彻底搞清楚JAVA的对象拷贝

    目录 一.背景 二.JAVA对象拷贝的实现 2.1 浅拷贝 2.2 深拷贝的实现方法一 2.3 深拷贝的实现方法二 2.3.1 C++拷贝构造函数 2.3.2 C++源码 2.3.3 JAVA通过拷贝 ...

  5. java bean对象拷贝

    Java的bean的属性复制,大家可以都看一下. 谈谈Java开发中的对象拷贝http://www.wtnull.com/view/2/e6a7a8818da742758bcd8b73d49d6be2 ...

  6. JVM系列之:详解java object对象在heap中的结构

    目录 简介 对象和其隐藏的秘密 Object对象头 数组对象头 整个对象的结构 简介 在之前的文章中,我们介绍了使用JOL这一神器来解析java类或者java实例在内存中占用的空间地址. 今天,我们会 ...

  7. java 的对象拷贝(有深浅拷贝两种方式,深拷贝实现的两种方式(逐层实现cloneable接口,序列化的方式来实现))

    Java提高篇--对象克隆(复制)(转自:http://www.cnblogs.com/Qian123/p/5710533.html#_label0)   阅读目录 为什么要克隆? 如何实现克隆 浅克 ...

  8. java Object对象的clone方法

    参考copy链接:http://blog.csdn.net/bigconvience/article/details/25025561 在看原型模式,发现要用到clone这个方法,以前和朋友聊过,没怎 ...

  9. 重写Java Object对象的hashCode和equals方法实现集合元素按内容判重

    Java API提供的集合框架中Set接口下的集合对象默认是不能存储重复对象的,这里的重复判定是按照对象实例句柄的地址来判定的,地址相同则判定为重复,地址不同不管内容如何都判定为不重复,这有时与需求不 ...

随机推荐

  1. Sara Chipps

    "To be a developer you really need to love to learn because you're going to be learning your en ...

  2. <hdu - 1272> 小希的迷宫 并查集问题 (注意特殊情况)

     本题链接:http://acm.hdu.edu.cn/showproblem.php?pid=1272 Problem Description: 上次Gardon的迷宫城堡小希玩了很久(见Probl ...

  3. 在ubuntu14.04上安装openstack mitaka

    最近在工作环境安装部署了juno版本,在GE口测试网络性能不太满意,发现mitaka版本支持ovs-dpdk,于是抽时间安装实验一番. 参考官网的安装文档,先准备将mitaka版本安装好再配置ovs. ...

  4. NASPhoto Station不只是储存的强大照片管理功能

    减少漫长的讨论时间,进而让你的艺术作品更符合客户需求.Photo Station 让你集中存储照片.随处分享及存取相簿并轻松收集朋友和客户反馈. 串流照片到大屏幕电视 DS photo 支援 Appl ...

  5. 《C++反汇编与逆向分析技术揭秘》——观察各种表达式的求值过程

    ---恢复内容开始--- 加法: 示例: 常量相加,则在编译期间就计算出两个常量相加后的结果,直接将这个结果参与运算,减少了运行期的计算.当有变量参与运算时,会先取出内存中的数据,放入通用寄存器中,再 ...

  6. webstorm for mac 破解步骤

    第一步:Web storm下载--从官网下载就可以 第二步:下载 - Java for OS X 2015-001--https://support.apple.com/kb/DL1572?viewl ...

  7. GrowingIO 2016 数据驱动增长大会—— 一起做增长英雄

    GrowingIO 2016 数据驱动增长大会,首次聚齐了增长黑客之父 Sean Ellis .世界前十位前沿数据科学家张溪梦等数十位中美顶尖增长实践者: 链家.点融网.Camera360.量化派.北 ...

  8. AnimatorController反向运动学IK

    通过使用反向运动学IK,我们可以根据需要控制角色身体某个特定部位进行旋转或移动,达到想要的一些效果,比如:在移动时,让一只脚带伤拖行:让手抬起去拿桌上的苹果:让脑袋一直面向我们的色像机,就像一直注视着 ...

  9. openwrt 更改默认主题

    BB 版本默认的主题为 bootstrap,想要修改为其他的主题,可以按照如下方法修改: 1.查看可以使用的主题. "make menuconfig" ---> " ...

  10. iOS混合应用开发入门

    原文出处: cocoacontrols   译文出处:魏志峰(@JeremyWei)   欢迎分享原创到伯乐头条 http://blog.jobbole.com/46554/ 介绍 上周(译者:原文成 ...