Java常用设计模式————原型模式(二)之深拷贝与浅拷贝
引言
clone顧名思義就是復制, 在Java語言中, clone方法被對象調用,所以會復制對象。所謂的復制對象,首先要分配一個和源對象同樣大小的空間,在這個空間中創建一個新的對象。那么在java語言中,有幾種方式可以創建對象呢??
1 使用new操作符創建一個對象?
2 使用clone方法復制一個對象?
那么這兩種方式有什么相同和不同呢? new操作符的本意是分配內存。程序執行到new操作符時, 首先去看new操作符后面的類型,因為知道了類型,才能知道要分配多大的內存空間。分配完內存之后,再調用構造函數,填充對象的各個域,這一步叫做對象的初始化,構造方法返回后,一個對象創建完畢,可以把他的引用(地址)發布到外部,在外部就可以使用這個引用操縱這個對象。而clone在第一步是和new相似的, 都是分配內存,調用clone方法時,分配的內存和源對象(即調用clone方法的對象)相同,然后再使用原對象中對應的各個域,填充新對象的域, 填充完成之后,clone方法返回,一個新的相同的對象被創建,同樣可以把這個新對象的引用發布到外部。【詳解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方式實現的拷貝,默認是采用淺拷貝的方式,即只拷貝調用clone方法的對象,而對象內部引用的成員變量對象則不會一同進行復制,因此,就算是將內部的對象再進行clone拷貝,依然會出現內部對象的內部引用對象沒有復制的問題。所以一般情況下,clone可以實現不徹底的深拷貝,無法實現徹底的深拷貝。
????? ? 通過Serializable實現深拷貝,會真正的實現整個對象的復制。
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類也一定要實現Serializable接口才可以,否則會產生NotSerializableException的運行時異常。
?
總結
以上是生活随笔為你收集整理的Java常用设计模式————原型模式(二)之深拷贝与浅拷贝的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: law是什么的缩写_Lawyer和Att
- 下一篇: python怎么切换中文键盘_pytho