java 深拷贝与浅拷贝机制详解

概要:

在Java中,拷贝分为深拷贝和浅拷贝两种。java在公共超类Object中实现了一种叫做clone的方法,这种方法clone出来的新对象为浅拷贝,而通过自己定义的clone方法为深拷贝。

(一)Object中clone方法

如果我们new出一个新对象,用一个声明去引用它,之后又用另一个声明去引用前一个声明,那么最后的结果是:这两个声明的变量将指向同一个对象,一处被改全部被改。如果我们想创建一个对象的copy,这个copy和对象的各种属性完全相同,而且修改这个copy和原对象毫无关系,那么这个时候我们就要用到clone方法。

package Clone;

import java.util.Date;

/**

*

* @author QuinnNorris

* java中的两种拷贝机制

*/

public class Clone {

/**

* @param args

* @throws CloneNotSupportedException

*/

public static void main(String[] args) throws CloneNotSupportedException {

// TODO Auto-generated method stub

ClassA valA = new ClassA(1, "old", new Date());

// 声明一个新的ClassA对象,我们不需要太关注ClassA的功能

ClassA valB = valA;

// 将valA引用的对象赋给valB

valA.setObject("new");

// 更改valA中的值,此时valB也被更改了,因为valA和valB指向同一个对象

valB = valA.clone();//通过clone方法制造副本

}

}

ClassA类中关于clone方法的重写部分:

//需要实现Cloneable接口

public class ClassA implements Cloneable {

public ClassA clone() throws CloneNotSupportedException {

return (ClassA) super.clone();//调用父类(Object)的clone方法

}

}

1.如何使用Object中clone方法的

有人总结使用clone方法的四条法则,我们一起分享一下:

为了获取对象的一份拷贝,我们可以利用Object类的clone()方法。

在派生类中覆盖基类的clone()方法,并声明为public。

在派生类的clone()方法中,调用super.clone()。

在派生类中实现Cloneable接口。

2.protected修饰的clone方法

在java.lang.Object的中,他将clone方法设置为protected修饰,这是很特殊的一种情况。protected的作用域是:包可见+可继承。之所以这样设置,是因为这个方法要返回的是克隆出来的对象,即clone方法要去克隆的类型是未知的,没有办法确定返回值的类型,自然只能让子孙后代来实现它重写它,为了能够让后代继承而又不过与张开,设置为了protected类型。

3.实现clone方法需要实现Cloneable接口

那么我们重写clone方法的时候为什么要去实现Cloneable接口呢?事实上,Cloneable接口是java中的一个标记接口,标记接口是指那些没有方法和属性的接口,他们存在只是为了让大家知道一些信息,而且在用:xxx instanceof Cloneable 的时候可以进行判断。Cloneable这个接口的出现就是为了让设计者知道要进行克隆处理了。如果一个对象需要克隆,但是没有实现(实际上,这里的“实现”换成“写上”更准确)Cloneable接口,那么会产生一个已检验异常。

4.实现clone方法需要调用父类的clone

我们为了达到复制一个和调用方法的这个对象一模一样的对象的目的,我们需要使用父类的clone方法,父类也以此类推,知道达到了Object的clone方法,那么Object的clone方法有什么用呢?API中是这样说的:

protected Object clone( ) throws CloneNotSupportedException创建并返回此对象的一个副本。

“副本”的准确含义可能依赖于对象的类。这样做的目的是,对于任何对象 x,

表达式: x.clone() != x为 true,表达式:x.clone().getClass() == x.getClass()也为 true,但这些并非必须要满足的要求。

一般情况下:

x.clone().equals(x)为 true,但这并非必须要满足的要求。

按照惯例,返回的对象应该通过调用 super.clone 获得。

如果一个类及其所有的超类(Object 除外)都遵守此约定,则 x.clone().getClass() == x.getClass()。

上面就是API中对clone的一部分基本讲解。我们可以得出结论的是,只要合理的调用了spuer.clone( )它就会返回一个被克隆的对象。在运行时刻,Object中的clone()识别出你要复制的是哪一个对象,然后为此对象分配空间,并进行对象的复制,将原始对象的内容一一复制到新对象的存储空间中。在这个克隆对象中,所有的属性都和被克隆的对象的属性相同,而这些相同的属性分为两种:

第一种 : 八大原始类型和不可变的对象(比如String)

第二种 : 其他类对象

对于第一种,clone方法将他们的值设置为原对象的值,没有任何问题。对于第二种,clone方法只是简单的将复制的新对象的引用指向原对象指向的引用,第二种的类对象会被两个对象修改。那么这个时候就涉及一个深浅拷贝的概念了。

(二)浅拷贝

浅拷贝:被拷贝对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象。换言之,浅复制仅仅复制所考虑的对象,而不复制它所引用的对象。

比如举个例子,一个类A中有另外一个类B类型的变量。在A重写clone函数调用super.clone的时候,创建的新对象和原来对象中的类B类型的变量是同一个,他们指向了同一个B的类型变量。如果在A中对B的变量做了修改,在新的拷贝出来的对象中B的变量也会被同样的修改。

请记住,直接调用super.clone实现的clone方法全部都是浅拷贝。

(三)深拷贝

深拷贝:被拷贝对象的所有变量都含有与原来的对象相同的值,除去那些引用其他对象的变量。那些引用其他对象的变量将指向被复制过的新对象,而不再是原有的那些被引用的对象。换言之,深复制把要复制的对象所引用的对象都复制了一遍。

通俗的说,如果说浅拷贝,开始的时候是两条线,如果在最后有一个其他类的变量,那么这两条线最后会合二为一,共同指向这变量,都能对他进行操作。深拷贝则是完完全全的两条线,互不干涉,因为他已经把所有的内部中的变量的对象全都复制一遍了。

深拷贝在代码中,需要在clone方法中多书写调用这个类中其他类的变量的clone函数。

(四)串行化深拷贝

在框架中,有的时候我们发现其中并没有重写clone方法,那么我们在需要拷贝一个对象的时候是如何去操作的呢?答案是我们经常会使用串行化方法,实现Serializable接口。

去寻找其他的方法来替代深拷贝也是无可奈何的事情,如果采用传统的深拷贝,难道你拷贝一个对象的时候向其中追无数层来拷贝完所有的对象变量么?先不谈这么做的时间消耗,仅仅是写这样的代码都会让人望而生畏。串行化深拷贝就是这样一个相对简单的方法。

把对象写到流里的过程是串行化(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());

}

虽然这种学院派的代码看起来很复杂,其实只是把对象放到流里,再拿出来。相比较分析判断无数的clone,这样简直是再简单不过了。这样做的前提是对象以及对象内部所有引用到的对象都是可串行化的,否则,就需要仔细考察那些不可串行化的对象是否设成transient。

transient:一个对象只要实现了Serilizable接口,这个对象就可以被序列化(序列化是指将java代码以字节序列的形式写出,即我们上面代码前三行写入对象),Java的这种序列化模式为开发者提供了很多便利,可以不必关系具体序列化的过程,只要这个类实现了Serilizable接口,这个的所有属性和方法都会自动序列化。但是有种情况是有些属性是不需要序列号的,所以就用到这个关键字。只需要实现Serilizable接口,将不需要序列化的属性前添加关键字transient,序列化对象的时候,这个属性就不会序列化到指定的目的地中。

(五)总结

在实际的应用中,深拷贝和浅拷贝只是两个概念,不一定谁比谁好,要按照实际的工作来确定如何去拷贝一个对象。如果在数据库操作方面,为了取出一张表时不涉及其他的表,肯定需要使用浅拷贝,而在框架的Serializable中,虽然耗时,但是深拷贝是非常有必要的。

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

java 深拷贝_java 深拷贝与浅拷贝机制详解相关推荐

  1. Java垃圾回收(GC)机制详解

    Java垃圾回收(GC)机制详解 转自:https://www.cnblogs.com/xiaoxi/p/6486852.html 一.为什么需要垃圾回收 如果不进行垃圾回收,内存迟早都会被消耗空,因 ...

  2. 反射 数据类型_Java基础:反射机制详解

    一.什么是反射: (1)Java反射机制的核心是在程序运行时动态加载类并获取类的详细信息,从而操作类或对象的属性和方法.本质是JVM得到class对象之后,再通过class对象进行反编译,从而获取t对 ...

  3. java 可见性_Java并发编程-volatile可见性详解

    前言 要学习好Java的多线程,就一定得对volatile关键字的作用机制了熟于胸.最近博主看了大量关于volatile的相关博客,对其有了一点初步的理解和认识,下面通过自己的话叙述整理一遍. 有什么 ...

  4. 【Java】反射( reflection)机制 详解

    目录 1. 定义 2. 用途(了解即可) 3. 反射的基本信息 4. 反射相关的类(重要) 4.1 Class类(反射机制的起源 ) 4.2 Class类中的相关方法(方法的使用方法在后边的示例当中) ...

  5. java 复合_Java复合语句的使用方法详解

    与 C 语言及其他语言相同, Java 语言的复合语句是以整个块区为单位的语句,所以又称为块语句.下面我们来看看有关复合语句的使用方法和实例. 复合语句由开括号"{"开始,闭括号& ...

  6. 希尔排序基础java代码_java 算法之希尔排序详解及实现代码

    摘要:这篇Java开发技术栏目下的"java 算法之希尔排序详解及实现代码",介绍的技术点是"希尔排序详解.实现代码.希尔排序.Java.实现.代码",希望对大 ...

  7. java中流_Java中流的有关知识点详解

    Java中流的有关知识点详解 发布时间:2020-09-17 03:50:59 来源:脚本之家 阅读:103 作者:mumu1998 什么是流? 流:程序和设备之间连接起来的一根用于数据传输的管道,流 ...

  8. java 搜索_Java实现搜索功能代码详解

    首先,我们要清楚搜索框中根据关键字进行条件搜索发送的是Get请求,并且是向当前页面发送Get请求 //示例代码 请求路径为当前页面路径 "/product" 当我们要实现多条件搜索 ...

  9. java 超时重试机制_Java之Retry重试机制详解

    应用中需要实现一个功能: 需要将 常规解决方案 try- 在包装正常上传逻辑基础上,通过判断返回结果或监听异常决定是否重试,同时为了解决立即重试的无效执行(假设异常是有外部执行不稳定导致的:网络抖动) ...

最新文章

  1. 如何使用echo.js实现图片的懒加载(整理)
  2. 人工智能:自由能理论,AI未来的数学模型
  3. python多线程为啥是假的?(GIL 全局解释器锁)(python多线程不适合并行化的计算密集型代码)
  4. 体验cas server
  5. Java每天5道面试题,跟我走,offer有!(九)
  6. Git中Add后对部分文件进行取消
  7. 10 分钟入门 Less 和 Sass
  8. 任晶磊:如何看待程序员在 GitHub 发起抗议互联网公司实行 996 工作制网站?
  9. 魔力宝贝服务器修改技能经验,传说中的技能及修改建议
  10. PostgreSQL-13-缺失值处理
  11. PDF417数据容量
  12. VS安装包注册com组件
  13. 使用openssl实现AES CBC 128 pcks7加密
  14. EXCEL批量删除行
  15. 机房怎么制作服务器,电信服务器机房服务器搬迁地网制作方法
  16. win10网络显示已连接到服务器异常,w10 网络连接配置异常如何修复
  17. 微信小程序、前端:背景图片样式拉伸不变形
  18. android 4.4 录屏方法,android 4.4 录屏方法
  19. 春节不断电之机器学习 —— 决策树
  20. win7台式计算机型号怎么查,win7怎么看电脑型号 win7电脑型号怎么查

热门文章

  1. Cascade EF-GAN: Progressive Facial Expression Editing with Local Focuses 论文解读
  2. Java分割由多个空白字符连接的字符串
  3. 【网络安全】XSS盲打实战案例:某网页漫画
  4. 【Web安全】通过机器学习破解验证码图片
  5. pwnable.kr lotto题解
  6. 012 背包二叉树遍历分析和代码编写
  7. 018 Android加固之实现dex加载器
  8. 008 Android之Service
  9. 160个Crackme045
  10. 【grafana】API 遇到的问题