网站首页 > java教程 正文
作者 | Java圣斗士 | 原创图文,转载请注明出处
全文2000字,阅读需要10分钟,建议收藏
哈喽大家好,我是又皮又可爱的Java圣斗士,关注我,每天带你飞!
我们昨天深入讨论了一下原型模式的使用场景以及代码实现,但是原型模式中有一个重要的问题可能会经常被问到,这个问题如果不彻底解决,那么原型模式也是一知半解。
通过昨天的学习,我们已经知道,重写Object类的clone()方法,并且实现Cloneable接口,就可以完成对象的克隆工作,减少创建对象时的开支。那么在比较克隆对象与原始对象的各个属性时,我们看到通过”==” 进行比较的结果是true。
public static void main(String[] args) { MyEntry con = new MyEntry("con", 23, new Part("part", 90)); con.show(); MyEntry clone = (MyEntry) con.clone(); clone.show(); System.out.println(con.getName() == clone.getName()); System.out.println(con.getNum() == clone.getNum()); System.out.println(con.getPart() == clone.getPart()); }
输出:
{name : con, num : 23, part : {partName : part, score : 90, inner : {innerName : inner, innerScore : 123}}} {name : con, num : 23, part : {partName : part, score : 90, inner : {innerName : inner, innerScore : 123}}} true true true
可以看到,不论是int类型、String类型还是封装类型Part,全都是true,而我们也都知道 “==” 与 “equals()”的区别,两者虽然都是判断对象是否相等,但前者是比较的物理内存地址,而后者才是真正语义上的比较。上面的代码中,name、num和part三个属性的地址全都是一样的,那么就说明:克隆对象和原始对象中的各个属性是同一份,这在对象拷贝中叫做浅拷贝,即不同的对象内部的属性指向相同的内存地址,如下图所示:
拷贝的分类
其实在Java中拷贝可以分为引用拷贝、对象拷贝,而对象拷贝又可以分为浅拷贝和深拷贝:
引用拷贝就是我们常用的赋值语句,不同的引用,指向同一个内存对象,这个很好理解:
而对象拷贝就是我们说的clone操作,而默认情况下,clone采用的是浅拷贝,这是因为在大部分情况下,人们往往需要的是内存中存储的内容,但要注意,在浅拷贝的情况下,如果对象中的属性有所变化,那么不同的类型,会有不同的表现形式:
public static void main(String[] args) { MyEntry con = new MyEntry("con", 23, new Part("part", 90)); con.show(); MyEntry clone = (MyEntry) con.clone(); clone.show(); clone.setName("clone"); clone.setNum(999); clone.getPart().setPartName("clonePart"); con.show(); clone.show(); }
输出:
{name : con, num : 23, part : {partName : part, score : 90, inner : {innerName : inner, innerScore : 123}}} {name : con, num : 23, part : {partName : part, score : 90, inner : {innerName : inner, innerScore : 123}}} {name : con, num : 23, part : {partName : clonePart, score : 90, inner : {innerName : inner, innerScore : 123}}} {name : clone, num : 999, part : {partName : clonePart, score : 90, inner : {innerName : inner, innerScore : 123}}}
深拷贝
深拷贝指的是将对象的所有属性全部拷贝一份,当克隆对象修改时,不论何种类型的属性,都不会影响到原型对象,在Java中,深拷贝的实现稍微复杂一些,所有需要依赖的封装类型变量全都需要实现Serializable接口,我们来看下面的代码:
protected Object clone() { try { // 序列化 ByteArrayOutputStream bos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(bos); oos.writeObject(this); // 反序列化 ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray()); ObjectInputStream ois = new ObjectInputStream(bis); return ois.readObject(); } catch (IOException | ClassNotFoundException e) { e.printStackTrace(); } return null; }
重写的clone()方法依赖了流的使用,通过Java自带的序列化机制,将对象以流的形式输出,并重新写入到一个新的内存区域中,实现完全克隆。
往期精彩:
《对话式情景剖析,String被final修饰的真正原因!一篇足矣》
---欢迎关注【Java圣斗士】,我是你们的小可爱(?ω?) Morty---
---专注IT职场经验、IT技术分享的灵魂写手---
---每天带你领略IT的魅力---
---期待与您陪伴!---
- 上一篇: java中浅拷贝和深拷贝的区别 #干货分享
- 下一篇: 一文了解Python深拷贝与浅拷贝问题
猜你喜欢
- 2024-10-19 Python 中赋值、浅拷贝、深拷贝的区别是什么?
- 2024-10-19 Java 对象拷贝原理剖析(java 对象拷贝原理剖析)
- 2024-10-19 认识Object类和深浅拷贝!(阐述object.assign的用法,深拷贝与浅拷贝的区别?)
- 2024-10-19 三面“有赞”Java岗斩获offer:Spring+JVM+并发锁+分布式+算法
- 2024-10-19 谈谈 Java 开发中的对象拷贝(java对象拷贝工具类)
- 2024-10-19 深入浅出Java中的clone克隆方法,写得也太棒了
- 2024-10-19 深拷贝和浅拷贝之list、dataframe
- 2024-10-19 对象拷贝java 浅谈(java对象的拷贝)
- 2024-10-19 Java克隆对象需要知道的事(java克隆的作用)
- 2024-10-19 Java的Object十二个常用方法,你用过几个?
你 发表评论:
欢迎- 最近发表
-
- 多种负载均衡算法及其Java代码实现
- 输入www.baidu.com背后经历了啥?说清楚这个,已经超过90%的人了
- 优化MySQL:为什么你应该用 UNSIGNED INT 存储IP地址
- 实模式下CPU如何获取数据及指令(实模式寻址方式)
- java基础都在这了,小主们拿去吧(java基础是指什么)
- 盘点爬虫语言为何选择Python而不是Java
- 搭载Dubbo+Zookeeper踩了这么多坑,我终于决定写下这篇
- 网络协议之TCP/IP协议(面试必考内容) - javaEE初阶 - 细节狂魔
- 深夜报警!10亿次请求暴击,如何用Redis找出最热IP?
- VPN技术(IPsec/L2TP/SSLVPN/PPTP)学习笔记
- 标签列表
-
- java反编译工具 (77)
- java反射 (57)
- java接口 (61)
- java随机数 (63)
- java7下载 (59)
- java数据结构 (61)
- java 三目运算符 (65)
- java对象转map (63)
- Java继承 (69)
- java字符串替换 (60)
- 快速排序java (59)
- java并发编程 (58)
- java api文档 (60)
- centos安装java (57)
- java调用webservice接口 (61)
- java深拷贝 (61)
- 工厂模式java (59)
- java代理模式 (59)
- java.lang (57)
- java连接mysql数据库 (67)
- java重载 (68)
- java 循环语句 (66)
- java反序列化 (58)
- java时间函数 (60)
- java是值传递还是引用传递 (62)
本文暂时没有评论,来添加一个吧(●'◡'●)