Java Object 对象拷贝
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 对象拷贝的更多相关文章
- Java Object 对象拷贝答疑
Java Object 对象拷贝答疑 @author ixenos 摘要:在对象的clone过程需要注意的几点.关于关键字this.super 关于clone[对象拷贝] 在实际编程过程,有时候我们会 ...
- Java Object 对象创建的方式 [ 转载 ]
Java Object 对象创建的方式 [ 转载 ] @author http://blog.csdn.net/mhmyqn/article/details/7943411 显式创建 有4种显式地创建 ...
- Java中对象拷贝的两种方式
引用的拷贝 //引用拷贝 private static void copyReferenceObject(){ Person p = new Person(23, "zhang") ...
- 通过与C++程序对比,彻底搞清楚JAVA的对象拷贝
目录 一.背景 二.JAVA对象拷贝的实现 2.1 浅拷贝 2.2 深拷贝的实现方法一 2.3 深拷贝的实现方法二 2.3.1 C++拷贝构造函数 2.3.2 C++源码 2.3.3 JAVA通过拷贝 ...
- java bean对象拷贝
Java的bean的属性复制,大家可以都看一下. 谈谈Java开发中的对象拷贝http://www.wtnull.com/view/2/e6a7a8818da742758bcd8b73d49d6be2 ...
- JVM系列之:详解java object对象在heap中的结构
目录 简介 对象和其隐藏的秘密 Object对象头 数组对象头 整个对象的结构 简介 在之前的文章中,我们介绍了使用JOL这一神器来解析java类或者java实例在内存中占用的空间地址. 今天,我们会 ...
- java 的对象拷贝(有深浅拷贝两种方式,深拷贝实现的两种方式(逐层实现cloneable接口,序列化的方式来实现))
Java提高篇--对象克隆(复制)(转自:http://www.cnblogs.com/Qian123/p/5710533.html#_label0) 阅读目录 为什么要克隆? 如何实现克隆 浅克 ...
- java Object对象的clone方法
参考copy链接:http://blog.csdn.net/bigconvience/article/details/25025561 在看原型模式,发现要用到clone这个方法,以前和朋友聊过,没怎 ...
- 重写Java Object对象的hashCode和equals方法实现集合元素按内容判重
Java API提供的集合框架中Set接口下的集合对象默认是不能存储重复对象的,这里的重复判定是按照对象实例句柄的地址来判定的,地址相同则判定为重复,地址不同不管内容如何都判定为不重复,这有时与需求不 ...
随机推荐
- 读取Oracle表结构数据
<html lang="zh-cn"> <head> <meta name="generator" content="O ...
- deployd使用归纳
deployd:一个生成后端数据的软件,简单的说就是大部分的前端不会后端,即使会也很难在深入到数据库进行设置一些前端所需数据的创建与查询的后端程序的书写,所以此时就是deployd大显身手的时候了. ...
- hdu1033
#include<stdio.h> #include<string.h> const int MAXN=200; char str[MAXN]; int main() { in ...
- 变形属性 transform
transform功能可以实现文字或图像的旋转.绽放.倾斜.与移动: 注意点:1.其移动.旋转.倾斜.与绽放这4种效果的使用先后顺序不同,页面会显示不同的结果: 2.属性值有一个参数与有多个参数的别: ...
- HDU 1013 Digital Roots(字符串)
Digital Roots Problem Description The digital root of a positive integer is found by summing the dig ...
- 动态多条件查询分页以及排序(一)--MVC与Entity Framework版url分页版
一.前言 多条件查询分页以及排序 每个系统里都会有这个的代码 做好这块 可以大大提高开发效率 所以博主分享下自己的6个版本的 多条件查询分页以及排序 二.目前状况 不论是ado.net 还是EF ...
- .Net中jQuery.ajax()调用asp.net后台方法 总结
利用JQuery的$.ajax()调用.Net后台方法有多种方式, 不多说了 直接上代码 前台代码 <script type="text/javascript"> $ ...
- 《C程序设计语言》 squeeze函数(从字符串s中删除字符c)
squeeze void squeeze(char string[], int ch) { int i, j; ; string[i] != '\0'; i++) { if (string[i] != ...
- Linux服务器性能指标查询命令安装
Linux命令扫盲 之 sar 今天在读<大规模Web服务开发技术>一书的时候,书中提到了sar这个命令,感觉很有用,有必要整理学习一下.(对于一位Linux初学者,不能放过任何一个学 ...
- HDU 4998 Rotate (计算几何)
HDU 4998 Rotate (计算几何) 题目链接http://acm.hdu.edu.cn/showproblem.php?pid=4998 Description Noting is more ...