龙空技术网

原型模式的深拷贝与浅拷贝,终于有人能够讲明白了

高级Bug调查员 73

前言:

如今朋友们对“浅拷贝原理”可能比较注重,朋友们都想要知道一些“浅拷贝原理”的相关资讯。那么小编也在网摘上收集了一些关于“浅拷贝原理””的相关知识,希望姐妹们能喜欢,你们一起来学习一下吧!

作者 | 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}}}truetruetrue

可以看到,不论是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参数传递的真正秘密》

《教你如何优雅地用Java8 实现日期时间的操作》

《每天扫描二维码,还不知道二维码登录原理的看过来》

《如何计算2 * 8?100%的面试官都想要这样的答案》

---欢迎关注【Java圣斗士】,我是你们的小可爱(✪ω✪) Morty---

---专注IT职场经验、IT技术分享的灵魂写手---

---每天带你领略IT的魅力---

---期待与您陪伴!---

标签: #浅拷贝原理