开发者社区> ghost丶桃子> 正文

java对象 深度克隆(不实现Cloneable接口)和浅度克隆

简介:
+关注继续查看

为什么需要克隆:

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

 

克隆的实现方式

一、浅度克隆

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

1、实现java.lang.Cloneable接口

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

 

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

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

 

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

 

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

 

 

 

Java代码

  1. public class Product implements Cloneable {   

  2.     private String name;   

  3.   

  4.     public Object clone() {   

  5.         try {   

  6.             return super.clone();   

  7.         } catch (CloneNotSupportedException e) {   

  8.             return null;   

  9.         }   

  10.     }   

  11. }  

 

 

 

 

二、深度克隆

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

 

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

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

 

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

 

 

 

Java代码

  1. public class Attribute implements Cloneable {   

  2.     private String no;   

  3.        

  4.     public Object clone() {   

  5.         try {   

  6.             return super.clone();   

  7.         } catch (CloneNotSupportedException e) {   

  8.             return null;   

  9.         }   

  10.     }   

  11. }   

  12.   

  13. public class Product implements Cloneable {   

  14.     private String name;   

  15.        

  16.     private Attribute attribute;   

  17.   

  18.     public Object clone() {   

  19.         try {   

  20.             return super.clone();   

  21.         } catch (CloneNotSupportedException e) {   

  22.             return null;   

  23.         }   

  24.     }   

  25. }  

  

 

 

 

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

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

 

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

 

虽然Java的序列化非常简单、强大,但是要用好,还有很多地方需要注意。比如曾经序列化了一个对象,可由于某种原因,该类做了一点点改动,然后重新被编译,那么这时反序列化刚才的对象,将会出现异常。 你可以通过添加serialVersionUID属性来解决这个问题。如果你的类是个单例(Singleton)类,是否允许用户通过序列化机制复制该类,如果不允许你需要谨慎对待该类的实现。

 

 

 

Java代码

  1. public class Attribute {   

  2.     private String no;   

  3. }   

  4.   

  5. public class Product {   

  6.     private String name;   

  7.        

  8.     private Attribute attribute;   

  9.   

  10.     public Product clone() {   

  11.         ByteArrayOutputStream byteOut = null;   

  12.         ObjectOutputStream objOut = null;   

  13.         ByteArrayInputStream byteIn = null;   

  14.         ObjectInputStream objIn = null;   

  15.            

  16.         try {   

  17.             byteOut = new ByteArrayOutputStream();    

  18.             objOut = new ObjectOutputStream(byteOut);    

  19.             objOut.writeObject(prototype);   

  20.   

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

  22.             objIn = new ObjectInputStream(byteIn);   

  23.                

  24.             return (ContretePrototype) objIn.readObject();   

  25.         } catch (IOException e) {   

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

  27.         } catch (ClassNotFoundException e) {   

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

  29.         } finally{   

  30.             try{   

  31.                 byteIn = null;   

  32.                 byteOut = null;   

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

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

  35.             }catch(IOException e){      

  36.             }      

  37.         }   

  38.     }   


特别说明:尊重作者的劳动成果,转载请注明出处哦~~~

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
Java用相同的方法在一个类中实现两个接口哪种接口方法被覆盖?
Java用相同的方法在一个类中实现两个接口哪种接口方法被覆盖?
16 0
Java开发——10.接口
Java中的接口更多的体现在对象行为的抽象!
9 0
java之WebService接口
java之WebService接口
16 0
【Java挠头】继承、抽象、接口、多态、向上转型、向下转型等精妙干货
【Java挠头】继承、抽象、接口、多态、向上转型、向下转型等精妙干货
16 0
java程序设计与j2ee中间件技术/软件开发技术(I)-实验三-接口、开闭原则和异常
java程序设计与j2ee中间件技术/软件开发技术(I)-实验三-接口、开闭原则和异常
32 0
Java集合Collection接口中的常用方法演示
由于接口不可以创建对象,所以,以下方法的演示将以Collection接口的子接口List的实现类ArrayList()进行演示。
19 0
java接口总结
接口声明:interface 接口的名字 接口体:接口体中包含常量的声明(没有变量)和抽象方法(没有普通方法)两部分。
29 0
【已解决】Java 项目中接入天翼云短信推送接口
【已解决】Java 项目中接入天翼云短信推送接口
21 0
文章
问答
文章排行榜
最热
最新
相关电子书
更多
Java工程师必读手册
立即下载
Java应用提速(速度与激情)
立即下载
Java单元测试实战
立即下载