日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > java >内容正文

java

Java常用设计模式————原型模式(二)之深拷贝与浅拷贝

發(fā)布時間:2025/3/12 java 26 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Java常用设计模式————原型模式(二)之深拷贝与浅拷贝 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

引言

clone顧名思義就是復制, 在Java語言中, clone方法被對象調用,所以會復制對象。所謂的復制對象,首先要分配一個和源對象同樣大小的空間,在這個空間中創(chuàng)建一個新的對象。那么在java語言中,有幾種方式可以創(chuàng)建對象呢??

1 使用new操作符創(chuàng)建一個對象?

2 使用clone方法復制一個對象?

那么這兩種方式有什么相同和不同呢? new操作符的本意是分配內存。程序執(zhí)行到new操作符時, 首先去看new操作符后面的類型,因為知道了類型,才能知道要分配多大的內存空間。分配完內存之后,再調用構造函數(shù),填充對象的各個域,這一步叫做對象的初始化,構造方法返回后,一個對象創(chuàng)建完畢,可以把他的引用(地址)發(fā)布到外部,在外部就可以使用這個引用操縱這個對象。而clone在第一步是和new相似的, 都是分配內存,調用clone方法時,分配的內存和源對象(即調用clone方法的對象)相同,然后再使用原對象中對應的各個域,填充新對象的域, 填充完成之后,clone方法返回,一個新的相同的對象被創(chuàng)建,同樣可以把這個新對象的引用發(fā)布到外部。【詳解Java中的clone方法】

拷貝的種類

引用拷貝

package design.pattern.copy; /*** 引用拷貝* <br>類名:CiteCopy<br>* 作者: mht<br>* 日期: 2018年4月1日-下午6:13:37<br>*/ public class CiteCopy {public static void main(String[] args) {Student s1 = new Student();Student s2 = s1;System.out.println(s1 + "\n" + s2);} }

結果:

design.pattern.copy.Student@6d06d12f design.pattern.copy.Student@6d06d12f


?

對象拷貝

?

package design.pattern.copy; /*** 對象拷貝* <br>類名:Copy<br>* 作者: mht<br>* 日期: 2018年4月1日-下午6:13:37<br>*/ public class Copy {public static void main(String[] args) {Student s1 = new Student();Student s2 = (Student) s1.clone();System.out.println(s1 + "\n" + s2);} }

結果:

design.pattern.copy.Student@15db9742 design.pattern.copy.Student@6d06d69c

淺拷貝

package design.pattern.copy; /*** 淺拷貝* <br>類名:Copy<br>* 作者: mht<br>* 日期: 2018年4月1日-下午6:13:37<br>*/ public class Copy {public static void main(String[] args) {Student s1 = new Student();Teacher t1 = new Teacher();s1.setTeacher(t1);System.out.println("t1的內存地址:" + t1);Student s2 = (Student) s1.clone();// 兩個學生對象的內存為:System.out.println(s1 + "\n" + s2);// 兩個學生對象中的老師對象為:System.out.println(s1.getTeacher() + "\n" + s2.getTeacher());} }

結果:

t1的內存地址:design.pattern.copy.Teacher@15db9742 design.pattern.copy.Student@6d06d69c design.pattern.copy.Student@7852e922 design.pattern.copy.Teacher@15db9742 design.pattern.copy.Teacher@15db9742

?

深拷貝

?

????????通過clone方式實現(xiàn)的拷貝,默認是采用淺拷貝的方式,即只拷貝調用clone方法的對象,而對象內部引用的成員變量對象則不會一同進行復制,因此,就算是將內部的對象再進行clone拷貝,依然會出現(xiàn)內部對象的內部引用對象沒有復制的問題。所以一般情況下,clone可以實現(xiàn)不徹底的深拷貝,無法實現(xiàn)徹底的深拷貝。

????? ? 通過Serializable實現(xiàn)深拷貝,會真正的實現(xiàn)整個對象的復制。

package design.pattern.copy;import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.io.Serializable;public class Student implements Serializable{private String name;private int age;private Teacher teacher;public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public Teacher getTeacher() {return teacher;}public void setTeacher(Teacher teacher) {this.teacher = teacher;}/*** 深拷貝* <br>作者: mht<br> * 時間:2018年4月1日-下午10:53:41<br>* @return*/protected Object deepClone(){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 e) {e.printStackTrace();} catch (ClassNotFoundException e) {e.printStackTrace();}return null;} } package design.pattern.copy;import java.io.Serializable;public class Teacher implements Serializable{private String name;private int age;public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;} } package design.pattern.copy; /*** Serializable序列化深拷貝* <br>類名:Copy<br>* 作者: mht<br>* 日期: 2018年4月1日-下午6:13:37<br>*/ public class Copy {public static void main(String[] args) {Student s1 = new Student();Teacher t1 = new Teacher();s1.setTeacher(t1);System.out.println("t1的內存地址:" + t1);System.out.println("==========================");Student s2 = (Student) s1.deepClone();// 兩個學生對象的內存為:System.out.println(s1 + "\n" + s2);System.out.println("==========================");// 兩個學生對象中的老師對象為:System.out.println("s1.getTeacher() : " + s1.getTeacher() + "\ns2.getTeacher() : " + s2.getTeacher());} }

結果:

t1的內存地址:design.pattern.copy.Teacher@15db9742 ========================== design.pattern.copy.Student@3d4eac69 design.pattern.copy.Student@232204a1 ========================== s1.getTeacher() : design.pattern.copy.Teacher@15db9742 s2.getTeacher() : design.pattern.copy.Teacher@4aa298b7

從上述結果可以看出,不僅student對象進行了復制,其s1內部的teacher對象也一同被復制了一個新的對象出來。但是要注意的是,Teacher類也一定要實現(xiàn)Serializable接口才可以,否則會產(chǎn)生NotSerializableException的運行時異常。

?

總結

以上是生活随笔為你收集整理的Java常用设计模式————原型模式(二)之深拷贝与浅拷贝的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內容還不錯,歡迎將生活随笔推薦給好友。