成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java中對象的深復制和淺復制詳解

開發 后端
這樣做的前提是對象以及對象內部所有引用到的對象都是可串行化的,否則,就需要仔細考察那些不可串行化的對象或屬性可否設成transient,從而將之排除在復制過程之外。上例代碼改進如下。

1.淺復制與深復制概念

⑴淺復制(淺克隆)

被復制對象的所有變量都含有與原來的對象相同的值,而所有的對其他對象的引用仍然指向原來的對象。換言之,淺復制僅僅復制所考慮的對象,而不復制它所引用的對象。

⑵深復制(深克隆)

被復制對象的所有變量都含有與原來的對象相同的值,除去那些引用其他對象的變量。那些引用其他對象的變量將指向被復制過的新對象,而不再是原有的那些被引用的對象。換言之,深復制把要復制的對象所引用的對象都復制了一遍。

2.Java的clone()方法

⑴clone方法將對象復制了一份并返回給調用者。一般而言,clone()方法滿足:

①對任何的對象x,都有x.clone() !=x//克隆對象與原對象不是同一個對象
②對任何的對象x,都有x.clone().getClass()= =x.getClass()//克隆對象與原對象的類型一樣
③如果對象x的equals()方法定義恰當,那么x.clone().equals(x)應該成立。

⑵Java中對象的克隆

①為了獲取對象的一份拷貝,我們可以利用Object類的clone()方法。
②在派生類中覆蓋基類的clone()方法,并聲明為public。
③在派生類的clone()方法中,調用super.clone()。
④在派生類中實現Cloneable接口。

請看如下代碼:

  1. public class Student implements Cloneable 
  2.   String name; 
  3. int age; 
  4.   Student(String name,int age) 
  5.   { 
  6.   this.name=name; 
  7.   this.age=age; 
  8.   } 
  9. public Object clone() 
  10.   { 
  11.    Object o=null
  12.   try 
  13.    { 
  14.    o=(Student)super.clone();//Object 中的clone()識別出你要復制的是哪一個對象。 
  15.    } 
  16.   catch(CloneNotSupportedException e) 
  17.    { 
  18.     System.out.println(e.toString()); 
  19.    } 
  20.   return o; 
  21.   }  
  22.  
  23. public static void main(String[] args) 
  24.   { 
  25.   Student s1=new Student("zhangsan",18); 
  26.   Student s2=(Student)s1.clone(); 
  27.   s2.name="lisi"
  28.   s2.age=20
  29.   //修改學生2后,不影響學生1的值。 
  30.   System.out.println("name="+s1.name+","+"age="+s1.age); 
  31.   System.out.println("name="+s2.name+","+"age="+s2.age); 

說明:

①為什么我們在派生類中覆蓋Object的clone()方法時,一定要調用super.clone()呢?在運行時刻,Object中的 clone() 識別出你要復制的是哪一個對象,然后為此對象分配空間,并進行對象的復制,將原始對象的內容一一復制到新對象的存儲空間中。

②繼承自java.lang.Object類的clone()方法是淺復制。以下代碼可以證明之。

  1. class Professor 
  2.   String name; 
  3.   int age; 
  4.   Professor(String name,int age) 
  5.   { 
  6.   this.name=name; 
  7.   this.age=age; 
  8.   } 
  9. public class Student implements Cloneable 
  10.   String name;// 常量對象。 
  11.   int age; 
  12.   Professor p;// 學生1和學生2的引用值都是一樣的。 
  13.   Student(String name,int age,Professor p) 
  14.   { 
  15.   this.name=name; 
  16.   this.age=age; 
  17.   this.p=p; 
  18.   } 
  19. public Object clone() 
  20.   { 
  21.    Student o=null
  22.   try 
  23.    { 
  24.     o=(Student)super.clone(); 
  25.    } 
  26.   catch(CloneNotSupportedException e) 
  27.    { 
  28.     System.out.println(e.toString()); 
  29.    } 
  30.    o.p=(Professor)p.clone(); 
  31.   return o; 
  32.   }  
  33. public static void main(String[] args) 
  34.   Professor p=new Professor("wangwu",50); 
  35.   Student s1=new Student("zhangsan",18,p); 
  36.   Student s2=(Student)s1.clone(); 
  37.   s2.p.name="lisi"
  38.   s2.p.age=30;  
  39.   System.out.println("name="+s1.p.name+","+"age="+s1.p.age); 
  40.   System.out.println("name="+s2.p.name+","+"age="+s2.p.age); 
  41.   //輸出結果學生1和2的教授成為lisi,age為30。 
  42.   } 

那應該如何實現深層次的克隆,即修改s2的教授不會影響s1的教授?代碼改進如下。

改進使學生1的Professor不改變(深層次的克隆)

  1. class Professor implements Cloneable 
  2.   String name; 
  3.   int age; 
  4.   Professor(String name,int age) 
  5.   { 
  6.   this.name=name; 
  7.   this.age=age; 
  8.   } 
  9. public Object clone() 
  10.   { 
  11.    Object o=null
  12.   try 
  13.    { 
  14.     o=super.clone(); 
  15.    } 
  16.   catch(CloneNotSupportedException e) 
  17.    { 
  18.     System.out.println(e.toString()); 
  19.    } 
  20.   return o; 
  21.   } 
  22. public class Student implements Cloneable 
  23.   String name; 
  24.   int age; 
  25.   Professor p; 
  26.   Student(String name,int age,Professor p) 
  27.   { 
  28.   this.name=name; 
  29.   this.age=age; 
  30.   this.p=p; 
  31.   } 
  32. public Object clone() 
  33.   { 
  34.    Student o=null
  35.   try 
  36.    { 
  37.     o=(Student)super.clone(); 
  38.    } 
  39.   catch(CloneNotSupportedException e) 
  40.    { 
  41.     System.out.println(e.toString()); 
  42.    } 
  43.    //對引用的對象也進行復制 
  44.    o.p=(Professor)p.clone(); 
  45.   return o; 
  46.   }  
  47. public static void main(String[] args) 
  48.   { 
  49.   Professor p=new Professor("wangwu",50); 
  50.   Student s1=new Student("zhangsan",18,p); 
  51.   Student s2=(Student)s1.clone(); 
  52.   s2.p.name="lisi"
  53.   s2.p.age=30
  54.   //學生1的教授不 改變。 
  55.   System.out.println("name="+s1.p.name+","+"age="+s1.p.age); 
  56.   System.out.println("name="+s2.p.name+","+"age="+s2.p.age); 

3.利用串行化來做深復制(主要是為了避免重寫比較復雜對象的深復制的clone()方法,也可以程序實現斷點續傳等等功能)

把對象寫到流里的過程是串行化(Serilization)過程,但是在Java程序師圈子里又非常形象地稱為“冷凍”或者“腌咸菜 (picking)” 過程;而把對象從流中讀出來的并行化(Deserialization)過程則叫做 “解凍”或者“回鮮(depicking)”過程。

應當指出的是,寫在流里的是對象的一個拷貝,而原對象仍然存在于JVM里面,因此“腌成咸菜”的只是對象的一個拷貝,Java咸菜還可以回鮮。

在Java語言里深復制一個對象,常常可以先使對象實現Serializable接口,然后把對象(實際上只是對象的一個拷貝)寫到一個流里(腌成咸菜),再從流里讀出來(把咸菜回鮮),便可以重建對象。

如下為深復制源代碼。

  1. public Object deepClone() 
  2. //將對象寫到流里 
  3. ByteArrayOutoutStream bo=new ByteArrayOutputStream(); 
  4. ObjectOutputStream oo=new ObjectOutputStream(bo); 
  5. oo.writeObject(this); 
  6. //從流里讀出來 
  7. ByteArrayInputStream bi=new ByteArrayInputStream(bo.toByteArray()); 
  8. ObjectInputStream oi=new ObjectInputStream(bi); 
  9. return(oi.readObject()); 

這樣做的前提是對象以及對象內部所有引用到的對象都是可串行化的,否則,就需要仔細考察那些不可串行化的對象或屬性可否設成transient,從而將之排除在復制過程之外。上例代碼改進如下。

 

  1. class Teacher implements Serializable{ 
  2.   String name; 
  3.   int age; 
  4.   public void Teacher(String name,int age){ 
  5.   this.name=name; 
  6.   this.age=age; 
  7.   } 
  8. public class Student implements Serializable{ 
  9. String name;//常量對象 
  10. int age; 
  11. Teacher t;//學生1和學生2的引用值都是一樣的。 
  12. public void Student(String name,int age,Teacher t){ 
  13.   this.name=name; 
  14.   this.age=age; 
  15.   this.p=p; 
  16. public Object deepClone() throws IOException, 
  17.     OptionalDataException,ClassNotFoundException{//將對象寫到流里 
  18.   ByteArrayOutoutStream bo=new ByteArrayOutputStream(); 
  19.   ObjectOutputStream oo=new ObjectOutputStream(bo); 
  20.   oo.writeObject(this);//從流里讀出來 
  21.   ByteArrayInputStream bi=new ByteArrayInputStream(bo.toByteArray()); 
  22.   ObjectInputStream oi=new ObjectInputStream(bi); 
  23.   return(oi.readObject()); 
  24. public static void main(String[] args){ 
  25.   Teacher t=new Teacher("tangliang",30); 
  26.   Student s1=new Student("zhangsan",18,t); 
  27.   Student s2=(Student)s1.deepClone(); 
  28.   s2.t.name="tony"
  29.   s2.t.age=40
  30.   //學生1的老師不改變 
  31.   System.out.println("name="+s1.t.name+","+"age="+s1.t.age); 
  32. }
責任編輯:王雪燕 來源: 陶邦仁
相關推薦

2025-04-27 09:45:58

JavaScript深拷貝淺拷貝

2010-07-27 15:03:37

Flex ArrayC

2023-05-05 08:47:35

Java淺拷貝深拷貝

2017-11-15 08:52:18

軟件硬件復制

2018-04-03 13:10:27

Java對象克隆

2022-09-12 07:59:13

操作系統LVM模式

2024-12-04 06:00:00

C#深拷貝

2024-04-11 08:30:05

JavaScript數組函數

2021-03-19 11:33:42

MySQL數據庫備份

2009-09-02 13:15:23

C#數組復制

2024-03-01 18:33:59

MySQL節點數據

2021-06-08 07:48:27

MySQL主從配置

2021-12-06 09:43:01

鏈表節點函數

2024-07-04 08:00:24

2010-08-12 15:30:10

MySQL集群

2010-03-23 16:30:47

Python文件復制

2023-11-07 18:03:31

Vim復制粘貼

2023-12-25 08:02:09

2024-11-19 13:11:19

2011-07-19 17:24:31

Objective-C 對象
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产精品不卡一区 | 欧美久久视频 | 精品国产乱码一区二区三区 | 9999国产精品欧美久久久久久 | 亚洲精品一区二区三区蜜桃久 | 日韩中文一区 | 91 在线 | 免费看片在线播放 | 天天操天天摸天天爽 | 祝你幸福电影在线观看 | 欧美日韩在线观看一区 | 中文字幕第十页 | 日韩在线视频一区二区三区 | 亚洲 欧美 日韩在线 | 国产精品久久久久久久久大全 | 亚洲成av人片在线观看无码 | 日韩视频在线免费观看 | 日韩在线免费 | 国内自拍偷拍 | 毛片在线看片 | 91在线看片 | 99精品欧美一区二区三区 | 午夜影院普通用户体验区 | 成人在线视频免费看 | 99久久婷婷国产综合精品电影 | 欧美激情亚洲激情 | 在线观看免费福利 | 天天天天天天操 | 精品一区二区三区四区视频 | 国产一区二区三区不卡av | 一区二区在线 | 一区二区中文 | 在线免费亚洲视频 | 国产一区二区在线播放 | 欧美精品一区二区免费视频 | 亚洲在线免费观看 | 精品自拍视频 | 久久激情五月丁香伊人 | 特a毛片 | 国产第一亚洲 | 国产在线精品一区 |