为什么需要克隆:

在实际编程过程中,我们常常要遇到这种情况:有一个对象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属性来解决这个问题。如果你的类是个单例(Singleton)类,是否允许用户通过序列化机制复制该类,如果不允许你需要谨慎对待该类的实现。

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 {

byteOut = new ByteArrayOutputStream();

objOut = new ObjectOutputStream(byteOut);

objOut.writeObject(prototype);

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){

}

}

}

}

java浅度克隆_java对象 深度克隆(不实现Cloneable接口)和浅度克隆相关推荐

  1. java对象序列化克隆_JAVA 对象克隆和序列化

    先用一个例子来说明假克隆吧,也就是用"="之后的效果.. [运行结果]: [运行结果] 姓名:rollen年龄:20 姓名:hello world年龄:100 姓名:hello w ...

  2. java 如何实现对象克隆_Java对象克隆

    Java不提供克隆(复制)对象的自动机制.克隆对象意味着逐位复制对象的内容.要支持克隆操作,请在类中实现clone()方法.Object类中的clone()方法的声明如下: protected Obj ...

  3. java 子类克隆_Java对象克隆了解

    1.java接口中有一种空的接口叫标识接口,这种接口值起到标识作用. 2.要实现java对象克隆需要用到Object类的 protected Object clone() throws CloneNo ...

  4. java 对象克隆_JAVA对象克隆

    1> 为了获取对象的一份拷贝,我们可以利用Object类的clone()方法. 2> 在派生类中覆盖基类的clone(),并声明为public. 3> 在派生类的clone()方法中 ...

  5. java初始化实例化_Java对象的创建过程:类的初始化与实例化

    一.Java对象创建时机 我们知道,一个对象在可以被使用之前必须要被正确地实例化.在Java代码中,有很多行为可以引起对象的创建,最为直观的一种就是使用new关键字来调用一个类的构造函数显式地创建对象 ...

  6. java jol原理_Java对象布局(JOL)实现过程解析

    java对象布局JOL(java object layout),描述对象在堆内存的布局.如下图: 1.markword 固定长度8byte,描述对象的identityhashcode,分代年龄,锁信息 ...

  7. java clone方法_JAVA对象clone方法代码实例解析

    1.Cloneable接口 这个接口一个方法都没有,但是要使用clone方法,就必须实现这个接口,否则抛出CloneNotSupportedException异常 2.clone方法的返回值 先附上两 ...

  8. java c 流_java对象传输流C/S传输对象

    java对象传输流C/S传输对象 java中有一个对象输入输出流,它可以包装普通的字节流. 我用一个C/S模式来测试练习对象输入输出流. 创建对象输入输出流包装基本字节流时遇到了第一个问题: 服务器创 ...

  9. java守护锁_Java 对象锁-synchronized()与线程的状态与生命周期与守护进程

    synchronized(someObject){ //对象锁 } 一.对象锁 someObject 的使用说明: 1.对象锁的返还. 当synchronize()语句执行完成. 当synchroni ...

最新文章

  1. 不使用第三个变量,实现交换两个变量的值
  2. PS Material 漫谈 六: Material Availability Check
  3. Fedora 从 15.0 开始将修改以太网卡命名规则
  4. ORACLE数据库,数据量大,转移数据到备份表语句
  5. JDBC(一)——statement对象、PreparedStatement对象
  6. Java进阶高级程序员必备:深入浅出springboot+深入Java虚拟机
  7. 中国已消失的九所世界级大学
  8. 删除特定Dns后缀_DNS根服务器里存了世界上所有的域名吗?
  9. 二叉树的递归遍历算法c语言 数据结构,递归创建二叉树c语言实现+详细解释
  10. 写给大一大二大三还在迷惘中的学生
  11. 记录一次etcd报错: listen tcp xxx.xxx.xxx.xxx:2380:bind: cannot assign requested address
  12. wordpress建立二级导航菜单
  13. 最新微信公众平台源码
  14. Linux 编译 python3.7
  15. 第十章(项目沟通管理)知识点
  16. java楼盘管理系统_课内资源 - 基于Java面向对象开发的房屋中介管理系统
  17. 紫微星情大全系列之紫微星
  18. Android--万能自定义弹窗
  19. mstsc Windows局域网内远程桌面连接
  20. 中基鸿业投资理财你需要知道的五件事

热门文章

  1. SpringBoot集成Cache缓存(Ehcache缓存框架,注解方式)
  2. Resx 文件无效。未能加载 .RESX 文件中使用的类型 System.Collections.Generic.List`1请确保已在项目中添加了必需的引用。
  3. 什么是JDK,什么是JRE?JDK的安装和环境变量的配置
  4. c++使用单向链表存储一组有序数据_初试攻略丨计算机考研中数据结构知识点总结,硬核!...
  5. java session 持久化_自定义实现session持久化
  6. JVM调优(二)垃圾回收算法
  7. C#调用bat 不显示DOS窗口,禁止DOS窗口一闪而过
  8. eclipse/MyEclipse 日期格式、注释日期格式、时区问题[转]
  9. [Android官方API阅读]___System Permissions
  10. 26个要素,仅仅依靠百度打造成功网站