深克隆

利用串行化来做深复制

把对象写到流里的过程是串行化(Serilization)过程,但是在Java程序师圈子里又非常形象地称为“冷冻”或者“腌咸菜(picking)”过程;而把对象从流中读出来的并行化(Deserialization)过程则叫做“解冻”或者“回鲜(depicking)”过程。应当指出的是,写在流里的是对象的一个拷贝,而原对象仍然存在于JVM里面,因此“腌成咸菜”的只是对象的一个拷贝,Java咸菜还可以回鲜。

在Java语言里深复制一个对象,常常可以先使对象实现Serializable接口,然后把对象(实际上只是对象的一个拷贝)写到一个流里(腌成咸菜),再从流里读出来(把咸菜回鲜),便可以重建对象。

public Object deepClone()
{
//将对象写到流里
ByteArrayOutoutStream bo=new ByteArrayOutputStream();
ObjectOutputStream oo=new ObjectOutputStream(bo);
oo.writeObject(this);
//从流里读出来
ByteArrayInputStream bi=new ByteArrayInputStream(bo.toByteArray());
ObjectInputStream oi=new ObjectInputStream(bi);
return(oi.readObject());
}

  这样做的前提是对象以及对象内部所有引用到的对象都是可串行化的,否则,就需要仔细考察那些不可串行化的对象可否设成transient,从而将之排除在复制过程之外。上例代码改进如下

时间: 2024-10-07 06:30:13

深克隆的相关文章

java中的拷贝(二)深克隆

浅拷贝(Object类中的clone()方法)是指在拷贝对象时,对于基本数据类型的变量会重新复制一份,而对于引用类型的变量只是对引用进行拷贝. 深拷贝(或叫深克隆) 则是对对象及该对象关联的对象内容,都会进行一份拷贝. 实体类: package com.lin.test; import java.io.Serializable; /** * @athor tianlin * * 2015年6月28日 下午1:56:18 * **/ public class Dog implements Seri

【java开发系列】—— 深克隆和浅克隆

Java支持我们对一个对象进行克隆,通常用在装饰模式和原型模式中.那么什么是深克隆,什么是浅克隆呢. [浅克隆],通常只是对克隆的实例进行复制,但里面的其他子对象,都是共用的. [深克隆],克隆的时候会复制它的子对象的引用,里面所有的变量和子对象都是又额外拷贝了一份. 下面的两个例子可以很好的说明他们的区别: 首先看一下类图 Husband类有一个对wife的引用,当进行浅克隆的时,wife变量都会指向同一个Wife:而进行深克隆时,会指向不同的Wife.下面进行一下验证: [浅克隆] 1 pu

Java学习笔记——设计模式之六.原型模式(浅克隆和深克隆)

That there's some good in this world, Mr. Frodo. And it's worth fighting for. 原型模式(prototype),用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. 至于Object.clone()这里就不赘述了.文档看起来可能有些难懂,直接上代码反而更容易理解: 浅克隆: 1 package cn.happy.design_pattern._06prototype.shallowclone; 2 3 pub

浅谈Java中的浅克隆与深克隆

在程序开发中如果已经存在一个对象A,现在需要一个与A对象完全相同的对象B,并对B对象的属性值进行修改,但是A对象原有的属性值不能改变.这是,如果使用Java提供的对象赋值语句,修改B对象的属性值后,A对象的属性值也将被修改.那么此时就应该想到要用java的clone方法去实现. 此时会遇到两种情况:第一种情况是,该对象中所有属性都是基本类型没有引用类型,这时就可以只重写Cloneable接口的clone方法即可:第二种情况就是,该对象中有其他对象的引用类型,此时只是前克隆就会失效,比如下述代码:

DOM克隆操作(深克隆/浅克隆)

深克隆(克隆元素内文本节点加上所有后辈元素节点), 浅克隆(克隆元素本身,不克隆文本节点和后辈节点), 1. cloneNode()接受一个可选值为true或false的参数.True 表示克隆元素和它的所有子节点.False表示克隆元素但不包含它的子节点.通常,我们在实践中用true,我从来没有遇到过想要克隆一个节点但不包含它的子节点的情形. 2. cloneNode()不会克隆事件处理程序.所以每次你克隆一个节点,你不得不在克隆上重新定义事件处理程序.在ie用attachEvent给dom元

JAVA深克隆与浅克隆1

复制就是得到一个副本 克隆就是复制一个对象的复本.但一个对象中可能有基本数据类型,如:int,long,float    等,也同时含有非基本数据类型如(数组,集合等)被克隆得到的对象基本类型的值修改了,原对象的值不会改变.这种适合shadow clone(浅克隆). 但如果你要改变一个非基本类型的值时,原对象的值却改变了.比如一个数组,内存中只copy他的地址,而这个地址指向的值并没有copy,当clone时,两个地址指向了一个值,这样一旦这个值改变了,原来的值当然也变了,因为他们共用一个值.

[C#]浅克隆和深克隆的区别和在C#中的体现形式

前言:我们知道对象分引用类型和值类型. 浅克隆:复制一个现有对象,引用类型指向同一个内存块(string为最特殊的对象,这里当作值类型来看先) public class User { public int Age { get; set; } public string UserName { get; set; } public List<string> List { get; set; } public User ShallowCopy() { return this.MemberwiseCl

JAVA之路_假克隆、浅克隆、深克隆

一.JAVA假克隆 Java中,对于基本类型,可以用"="进行克隆,而对于引用类型却不能简单的使用"="进行克隆,这与JAVA的内存使用空间有关,JAVA在栈中保存基本类型和引用变量,在堆中保存对象.对于引用变量而言,使用"="将修改引用,而不是复制堆中的对象,此时两个引用对象将指向同一个对象,因此如果对一个变量修改则会修改另一个对象. public class Employee { private String name; private in

浅克隆与深克隆

浅克隆: public class Person implements Cloneable { private String name; private int age; private Teacher teacher; public Person(String name, int age) { this.name = name; this.age = age; } @Override public Person clone() throws CloneNotSupportedException

Java中的浅克隆(shallow clone)与深克隆(deep clone)

Summary 浅克隆与深克隆对于JavaSE来说,是个难度系数比较低的概念,但不应该轻视它. 假设一个场景:对于某个list,代码里并没有任何对其的直接操作,但里面的元素的属性却被改变了,这可能就涉及到这个概念. Description 浅克隆指仅copy对象位于栈内存中的引用(reference).copy后,新旧两个引用指向同一个堆内存对象(即同一内存区域),但是堆内存中实际的对象copy前后均只有一个.使用"==" operator比较二者的地址会返回true.(不同引用,同一