Java对象的浅克隆和深克隆

为什么需要克隆

在实际编程过程中,我们常常要遇到这种情况:有一个对象A,在某一时刻A中已经包含了一些有效值,此时可能会需要一个和A完全相同新对象B, 并且此后对B任何改动都不会影响到A中的值,也就是说,A与B是两个独立的对象,但B的初始值是由A对象确定的。在Java语言中,用简单的赋值语句是不 能满足这种需求的,要满足这种需求有很多途径。

    克隆的实现方式


  一、浅度克隆

 
  

浅度克隆对于要克隆的对象,对于其基本数据类型的属性,复制一份给新产生的对象,对于非基本数据类型的属性,仅仅复制一份引用给新产生的对象,即新产生的对象和原始对象中的非基本数据类型的属性都指向的是同一个对象。

浅度克隆步骤:

1、实现java.lang.Cloneable接口

要clone的类为什么还要实现Cloneable接口呢?Cloneable接口是一个标识接口,不包含任何方法的!这个标识仅仅是针对Object类中clone()方法的,如果clone类没有实现Cloneable接口,并调用了Object的
clone()方法(也就是调用了super.Clone()方法),那么Object的clone()方法就会抛出
CloneNotSupportedException异常。

2、重写java.lang.Object.clone()方法

JDK
API的说明文档解释这个方法将返回Object对象的一个拷贝。要说明的有两点:一是拷贝对象返回的是一个新对象,而不是一个引用。二是拷贝对象与用new操作符返回的新对象的区别就是这个拷贝已经包含了一些原来对象的信息,而不是对象的初始信息。

观察一下Object类的clone()方法是一个native方法,native方法的效率一般来说都是远高于java中的非native方法。这也解
释了为什么要用Object中clone()方法而不是先new一个类,然后把原始对象中的信息赋到新对象中,虽然这也实现了clone功能。Object类中的clone()还是一个protected属性的方法,重写之后要把clone()方法的属性设置为public。

Object类中clone()方法产生的效果是:先在内存中开辟一块和原始对象一样的空间,然后原样拷贝原始对象中的内容。对基本数据类型,这样
的操作是没有问题的,但对非基本类型变量,我们知道它们保存的仅仅是对象的引用,这也导致clone后的非基本类型变量和原始对象中相应的变量指向的是同
一个对象。

Java代码实例:

public class Product implements Cloneable {

private String name;

public Object clone() {

try {

return
super.clone();

} catch (CloneNotSupportedException e) {

return
null;

}

}

}

   二、深度克隆

   
 

在浅度克隆的基础上,对于要克隆的对象中的非基本数据类型的属性对应的类,也实现克隆,这样对于非基本数据类型的属性,复制的不是一份引用,即新产生的对象和原始对象中的非基本数据类型的属性指向的不是同一个对象

深度克隆步骤:

要克隆的类和类中所有非基本数据类型的属性对应的类

1、都实现java.lang.Cloneable接口

2、都重写java.lang.Object.clone()方法

Java代码实例:

public class Attribute implements Cloneable {

private String no;

public Object clone() {

try {

return
super.clone();

} catch (CloneNotSupportedException e) {

return
null;

}

}

}

public class Product implements Cloneable {

private String name;

private Attribute attribute;

public Object clone() {

try {

return
super.clone();

} catch (CloneNotSupportedException e) {

return
null;

}

}

}

   
  三、使用对象序列化和反序列化实现深度克隆

所谓对象序列化就是将对象的状态转换成字节流,以后可以通过这些值再生成相同状态的对象。

对象的序列化还有另一个容易被大家忽略的功能就是对象复制(Clone),Java中通过Clone机制可以复制大部分的对象,但是众所周知,Clone有深度Clone和浅度Clone,如果你的对象非常非常复杂,并且想实现深层
Clone,如果使用序列化,不会超过10行代码就可以解决。

虽然Java的序列化非常简单、强大,但是要用好,还有很多地方需要注意。比如曾经序列化了一个对象,可由于某种原因,该类做了一点点改动,然后重新被编译,那么这时反序列化刚才的对象,将会出现异常。
你可以通过添加serialVersionUID属性来解决这个问题(在Java序列化与反序列化学习(二):序列化接口说明 的标题三中有此说明)。如果你的类是个单例(Singleton)类,虽然我们多线程下的并发问题来控制单例,但是,是否允许用户通过序列化机制或者克隆来复制该类,如果不允许你需要谨慎对待该类的实现(让此类不用实现Serializable接口或Externalizable接口和Cloneable接口就行了)

Java代码实例:

public class
Attribute {

private String no;

}

public class
Product {

private String name;

private Attribute attribute;

public Product clone() {

ByteArrayOutputStream byteOut = null;

ObjectOutputStream objOut = null;

ByteArrayInputStream byteIn = null;

ObjectInputStream objIn = null;

try {

//
将该对象序列化成流,因为写在流里的是对象的一个拷贝,而原对象仍然存在于JVM里面。所以利用这个特性可以实现对象的深拷贝

byteOut =
new ByteArrayOutputStream();

objOut =
new ObjectOutputStream(byteOut);

objOut.writeObject(this);

// 将流序列化成对象

byteIn =
new ByteArrayInputStream(byteOut.toByteArray());

objIn =
new ObjectInputStream(byteIn);

return
(ContretePrototype) objIn.readObject();

} catch (IOException e) {

throw new
RuntimeException("Clone Object failed in IO.",e);

} catch (ClassNotFoundException e) {

throw new
RuntimeException("Class not found.",e);

} finally{

try{

byteIn = null;

byteOut = null;

if(objOut != null)
objOut.close();

if(objIn != null)
objIn.close();

}catch(IOException e){

}

}

}

}

#########################注意######################

Bean复制的几种框架中(Apache
BeanUtils、PropertyUtils,Spring BeanUtils,Cglib
BeanCopier)都是相当于克隆中的浅克隆。

1)spring包和Apache中的
BeanUtils   
 采用反射实现

Spring: void
copyProperties(Object source, Object target,String[]
ignoreProperties)

Apache:void  copyProperties(Object dest, Object
orig)

2)cglib包中的
 Beancopier   采用动态字节码实现

cglib: BeanCopier
create(Class source, Class target,boolean
useConverter)

例如:

BeanCopier copier
=BeanCopier.create(stuSource.getClass(), stuTarget.getClass(),
false);

copier.copy(stuSource,
stuTarget, null);

公司内部对用到的bean属性复制做了下性能分析:

cglib   BeanCopier  
15ms

 

Spring  BeanUtil  
   4031ms

 

apache BeanUtils    
 18514ms.

时间: 2024-10-26 08:13:32

Java对象的浅克隆和深克隆的相关文章

java基础入门-对象的浅克隆与深克隆

这里面涉及到两个类,一个是person类,一个是测试类test 首先我们说到的是浅克隆,对某个对象实施Clone时对其是一无所知的,它仅仅是简单地执行域对域的copy,如果是基本数据类型(int,float,char等)到没什么问题,基本遇上如string,Integer等不可变对象的时候也没有什么问题,但是如果遇上了date这个可变对象,或者是自己定义的可变对象,他只是简单的复制一下引用这些可变对象,而不是把这些可变对象再一次的复制 先上person类,这里面虽然是实现Cloneable接口,

Java对象的浅克隆

/**  * Java对象的浅克隆  * 在克隆对象时,如果对象的成员变量是基本类型,则使用浅克隆即可完成.  * 如果对象的成员变量包括可变引用类型,则需要深克隆  * 如果引用类型是不可变的,如String类,则不必进行深克隆  * 当需要克隆对象时,需要使用clone()方法,该方法将类中的各个域进行复制,  * 如果对于引用类型的域,这种操作就会有问题,因此成为浅克隆.  * 提供克隆功能的类需要实现Cloneable接口,否则会抛出异常  */ public class Address

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

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

java(30) - 对象浅克隆和深克隆

一.浅克隆和深克隆的概念:        1).浅克隆:又称为浅复制,被复制的对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象.换而言之,浅复制仅仅复制所考虑的对象,而不复制它所引用的对象. 2).深克隆:又称为深复制,被复制的对象的所有变量都含有与原来的对象相同的值,除去那些引用其他对象的变量.那些引用其他对象的变量将指向被复制过的新对象,而不再是原有的那些被引用的对象.换而言之,深复制就是把要复制的对象所引用的对象都复制了一遍. 二.浅克隆实现: <pre

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

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

Java浅克隆和深克隆

什么是浅克隆和深克隆? 克隆顾名思义就是,参照一个东西再做一个出来 浅克隆:直接复写Object的clone()方法,默认情况下8种基本数据类型和String都会进行深克隆,另外的其他引用类型为浅克隆(浅克隆:引用指向的是同一个对象) 深克隆:浅克隆中那另外的其他引用类型都让其变为深克隆 引用类型图解 浅克隆 public class Main { public static void main(String[] args) throws Exception { A a = new A(); A

java对象的克隆

java对象克隆方式主要有两种:浅克隆和深克隆 首先,不要把对象的克隆和对象的赋值搞混了,看下图 p2 = p1;就是赋值操作,赋值操作只是让被赋值对象指向之前对象的地址,实际上的物理内存是一块,而克隆操作的结果应该是两个对象分别指向内容相同的两块内存.如下就是克隆操作后的状态: 下面说浅克隆和深克隆: 深克隆和浅克隆的区别主要出现在类里有外部类对象时,如下,Person类中有Address类的对象 1 package cn.itcast.copy; 2 3 import java.io.Ser

深入理解Java对象的创建过程:类的初始化与实例化

"-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"> 深入理解Java对象的创建过程:类的初始化与实例化 - Rico's Blogs - 博客频道 - CSDN.NET Rico's Blogs 潜心修炼,成为一个更好的人. 目录视图 摘要视图 订阅 [活动]2017 CSDN博客专栏评选 &nbsp [5月书讯]流畅

(转)深入理解Java对象的创建过程

参考来源:http://blog.csdn.net/justloveyou_/article/details/72466416 摘要: 在Java中,一个对象在可以被使用之前必须要被正确地初始化,这一点是Java规范规定的.在实例化一个对象时,JVM首先会检查相关类型是否已经加载并初始化,如果没有,则JVM立即进行加载并调用类构造器完成类的初始化.在类初始化过程中或初始化完毕后,根据具体情况才会去对类进行实例化.本文试图对JVM执行类初始化和实例化的过程做一个详细深入地介绍,以便从Java虚拟机