为什么要克隆

为什么要使用克隆,这其实反映的是一个很现实的问题,假如我们有一个对象:

public class SimpleObject implements Cloneable{ private String str;  public SimpleObject() { System.out.println("Enter SimpleObject.constructor()"); } public String getStr() { return str; } public void setStr(String str) { this.str = str; }  public Object clone() throws CloneNotSupportedException { return super.clone(); }}

现在我写一段程序:

public static void main(String[] args){ SimpleObject so0 = new SimpleObject(); so0.setStr("111"); System.out.println("so0.getStr():" + so0.getStr()); SimpleObject so1 = so0; so1.setStr("222"); System.out.println("so0.getStr():" + so0.getStr()); System.out.println("so1.getStr():" + so1.getStr());}

运行结果其实很明显:

so0.getStr():111so0.getStr():222so1.getStr():222

Java底层使用C/C++实现的,"="这个运算符,如果左右两边都是对象引用的话,在Java中表示的将等号右边的引用赋值给等号左边的引用,二者指向的还是同一块内存,所以任何一个引用对内存的操作都直接反映到另一个引用上。

但是,现在我想拿这个so0的数据进行一些操作,不想改变原来so0中的内容,这时候就可以使用克隆了,它允许在堆中克隆出一块和原对象一样的对象,并将这个对象的地址赋予新的引用,这样,显然我对新引用的操作,不会影响到原对象。

当然,理解克隆,最好还是对Java内存区域有比较好的理解。

Cloneable接口和Object的clone()方法

Java中实现了Cloneable接口的类有很多,像我们熟悉的ArrayList、Calendar、Date、HashMap、Hashtable、HashSet、LinkedList等等。

还是那句话,对于不熟悉的接口、方法,第一反应一定是查询JDK API。

1、Cloneable接口

三句话总结:

(1)此类实现了Cloneable接口,以指示Object的clone()方法可以合法地对该类实例进行按字段复制

(2)如果在没有实现Cloneable接口的实例上调用Object的clone()方法,则会导致抛出CloneNotSupporteddException

(3)按照惯例,实现此接口的类应该使用公共方法重写Object的clone()方法,Object的clone()方法是一个受保护的方法

2、Object的clone()方法

创建并返回此对象的一个副本。对于任何对象x,表达式:

(1)x.clone() != x为true

(2)x.clone().getClass() == x.getClass()为true

(3)x.clone().equals(x)一般情况下为true,但这并不是必须要满足的要求

克隆实例

把上面例子的main函数修改一下:

public static void main(String[] args) throws Exception{ SimpleObject so0 = new SimpleObject(); so0.setStr("111"); SimpleObject so1 = (SimpleObject)so0.clone();  System.out.println("so0 == so1?" + (so0 == so1)); System.out.println("so0.getClass() == so1.getClass()?" + (so0.getClass() == so1.getClass())); System.out.println("so0.equals(so1)?" + (so0.equals(so1)));  so1.setStr("222"); System.out.println("so0.getStr():" + so0.getStr()); System.out.println("so1.getStr():" + so1.getStr());}

看一下运行结果:

Enter SimpleObject.constructor()so0 == so1?falseso0.getClass() == so1.getClass()?trueso0.equals(so1)?falseso0.getStr():111so1.getStr():222

得到三个结论:

1、克隆一个对象并不会调用对象的构造方法,因为"Enter SimpleObject.constructor()"语句只出现了一次

2、符合JDK API的clone()方法三条规则

3、so1对于SimpleObject对象str字段的修改再也不会影响到so0了

浅克隆和深克隆

浅克隆(shallow clone)和深克隆(deep clone)反映的是,当对象中还有对象的时候,那么:

1、浅克隆,即很表层的克隆,如果我们要克隆对象,只克隆它自身以及它所包含的所有对象的引用地址

2、深克隆,克隆除自身对象以外的所有对象,包括自身所包含的所有对象实例

这两个概念应该很好理解,就不写代码了。多提一句,所有的基本数据类型,无论是浅克隆还是深克隆,都会进行原值克隆,毕竟它们都不是对象,不是存储在堆中的。

那其实Object的clone()方法,提供的是一种浅克隆的机制,如果想要实现对对象的深克隆,在不引入第三方jar包的情况下,可以使用两种办法:

1、先对对象进行序列化,紧接着马上反序列化出

2、先调用super.clone()方法克隆出一个新对象来,然后在子类的clone()方法中手动给克隆出来的非基本数据类型(引用类型)赋值,比如ArrayList的clone()方法:

public Object clone() {try { ArrayList v = (ArrayList) super.clone(); v.elementData = Arrays.copyOf(elementData, size); v.modCount = 0; return v;} catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable throw new InternalError();}}

java clone方法_Java基础:Cloneable接口和Object的clone()方法相关推荐

  1. 测试私有方法_Java基础之抽象类、接口作为方法参数和返回值

    不同修饰符使用细节 常用来修饰类.方法.变量的修饰符 public 权限修饰符,公共访问, 类,方法,成员变量 protected 权限修饰符,受保护访问, 方法,成员变量 默认什么也不写 也是一种权 ...

  2. 在java中班级的表示方法_Java基础班学习笔记(8)

    1:如何制作帮助文档(了解) (1)写一个类 (2)加入文档注释 (3)通过javadoc工具生成即可 javadoc -d 目录 -author -version ArrayTool.java 如何 ...

  3. java反射三种方法_Java基础入门要学哪些 怎么掌握反射和枚举

    Java基础入门要学哪些?怎么掌握反射和枚举?Java是老牌编程语言,是一种可以撰写跨平台应用软件的面向对象的程序设计语言.Java所包含的基础知识点很多,只有掌握了这些知识才能更好地运用,下面就来给 ...

  4. Java中introduce方法_Java基础—继承

    继承是面向对象的核心特征之一,是由已有类创建新类的机制.利用继承机制,可以先创建一个具有共性的一般类,然后根据该一般类创建具有特殊性的新类,新类继承一般类的属性和方法,并根据需要增加自己的新属性和方法 ...

  5. java 抽象类语法_JAVA基础语法8--多态/抽象类/抽象方法

    多态 继承.封装.多态.抽象是面向对象编程的四大基本特征.封装隐藏了类的内部实现机制,从而可以在不影响使用者的前提条件下,改变类的内部结构,同时保护了数据.继承是为了重用父类代码,同时为多态做准备.那 ...

  6. java super用法_Java基础面试题汇总

    blog.csdn.net/ThinkWon/article/details/104390612 Java概述 何为编程 编程就是让计算机为解决某个问题而使用某种程序设计语言编写程序代码,并最终得到结 ...

  7. java 5 多线程_Java基础系列五、多线程

    1.什么是进程?什么是线程? 进程概念:在计算机中运行的软件,是操作系统中最基础的组成部分 .进程是容器,里面装的都是线程. 线程概念:就是运行在进程中的一段代码,是进程中最小组织单元. 注意: 1. ...

  8. java list 差集_Java基础之集合框架

    Java 集合框架概述 一方面, 面向对象语言对事物的体现都是以对象的形式,为了方便对多个对象的操作,就要对对象进行存储.另一方面,使用Array存储对象方面具有一些弊端,而Java 集合就像一种容器 ...

  9. java调用驱动_Java中间件及其驱动调用、加载方法及转换装置的制造方法

    Java中间件及其驱动调用.加载方法及转换装置的制造方法 [技术领域] [0001]本发明涉及移动通信技术领域,尤其涉及Java中间件及其驱动调用.加载方法及转换装置. [背景技术] [0002]Ja ...

  10. java display.getdefault()_java基础(十一 )-----反射——Java高级开发必须懂的

    本文我们通过一个实际的例子来演示反射在编程中的应用,可能之前大家对反射的学习,仅仅是停留在概念层面,不知道反射究竟应用在哪,所以是一头雾水.相信通过这篇教程,会让你对反射有一个更深层次的认知. 概念 ...

最新文章

  1. 基于Nginx实现10万+并发,你应该做的Linux内核优化
  2. yl335b分拣站单元流程图_选择单元化物流容器的必要性
  3. SQLServer中char、varchar、nchar、nvarchar的区别:
  4. 硬盘的原理以及SQL Server如何利用硬盘原理减少IO
  5. 前端学习(2029)vue之电商管理系统电商系统之timeline组件
  6. UVa 12169 - Disgruntled Judge(拓展欧几里德)
  7. 使用Visual Studio 部署SharePoint时提示“路径中具有非法字符”
  8. 不同库表数据库迁移工具_Microsoft提供的数据库迁移助手工具概述
  9. 谷歌编译器收藏网站在哪打开
  10. 【Python】调用WPS V9 API,实现Word转PDF
  11. chronodex怎么用_滴答清单使用全攻略:如何把手帐搬到滴答清单上,提升效率?...
  12. mysql flush explain_Mysql_mysql 性能分析及explain用法
  13. Apache JMeter 5.1.1 Win 10 环境变量配置
  14. Animal Faces| 动物面部 | 数据集
  15. AD转换精度提高方法
  16. CAD角度标注命令,标注CAD图纸
  17. Java27岁啦——一次争执引起的Java内卷生涯
  18. Tarena代码-一些代码碎片
  19. 亲身经历灵魂附体与出马仙之说
  20. mysql-日志分析

热门文章

  1. 6.SOA架构:服务和微服务分析及设计--- Web服务的服务API与契约设计
  2. 44.Linux/Unix 系统编程手册(下) -- 管道和 FIFO
  3. 42. netcat
  4. 155.PHP中“==”运算符的安全问题
  5. 20.经典抽象数据类型
  6. 110. PHP 读取 ini ,ftp 上传
  7. css中的外边距合并时垂直方向上的普通流相邻元素间
  8. phpcms站点域名配置https无法提交如何处理
  9. .net core平台使用遇到的坑
  10. VB INET控件的全部用法