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

Java技巧:深拷貝的兩種方式

開發 后端
拷貝構造函數,經常被稱作X(X&),是一種特殊的構造函數,他由編譯器調用來完成一些基于同一類的其他對象的構件及初始化。它的唯一的一個參數(對象的引用)是不可變的(因為是const型的)。

⑴淺復制(淺克隆)

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

⑵深復制(深克隆)

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

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

 

說明:

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

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

 

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

 

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

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

 

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

 

3.利用串行化來做深復制

把對象寫到流里的過程是串行化(Serilization)過程,但是在Java程序師圈子里又非常形象地稱為“冷凍”或者“腌咸菜(picking)”過程;而把對象從流中讀出來的并行化(Deserialization)過程則叫做“解凍”或者“回鮮(depicking)”過程。應當指出的是,寫在流里的是對象的一個拷貝,而原對象仍然存在于JVM里面,因此“腌成咸菜”的只是對象的一個拷貝,Java咸菜還可以回鮮。

在Java語言里深復制一個對象,常??梢韵仁箤ο髮崿FSerializable接口,然后把對象(實際上只是對象的一個拷貝)寫到一個流里(腌成咸菜),再從流里讀出來(把咸菜回鮮),便可以重建對象。

如下為深復制源代碼。

 

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

 

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

 

  1. class Professor implements Serializable   
  2. {   
  3.     String name;   
  4.     int age;   
  5.     Professor(String name,int age)   
  6.     {   
  7.         this.name=name;   
  8.         this.age=age;   
  9.     }   
  10. }   
  11. class Student implements Serializable   
  12. {   
  13.     String name;//常量對象。   
  14.     int age;   
  15.     Professor p;//學生1和學生2的引用值都是一樣的。   
  16.     Student(String name,int age,Professor p)   
  17.     {   
  18.         this.name=name;   
  19.         this.age=age;   
  20.         this.p=p;   
  21.     }   
  22.     public Object deepClone() throws IOException,   
  23. OptionalDataException,ClassNotFoundException   
  24. {   
  25. //將對象寫到流里   
  26. ByteArrayOutoutStream bo=new ByteArrayOutputStream();   
  27. ObjectOutputStream oo=new ObjectOutputStream(bo);   
  28. oo.writeObject(this);   
  29. //從流里讀出來   
  30. ByteArrayInputStream bi=new ByteArrayInputStream(bo.toByteArray());   
  31. ObjectInputStream oi=new ObjectInputStream(bi);   
  32. return(oi.readObject());   
  33. }   
  34.  
  35. }   
  36. public static void main(String[] args)   
  37.     {   
  38.       Professor p=new Professor("wangwu",50);   
  39.       Student s1=new Student("zhangsan",18,p);   
  40.       Student s2=(Student)s1.deepClone();   
  41.       s2.p.name="lisi";   
  42.      s2.p.age=30;   
  43. System.out.println("name="+s1.p.name+","+"age="+s1.p.age); //學生1的教授不改變。   
  44. }  

【編輯推薦】

  1. Java程序開發中的簡單內存分析
  2. Java中靜態數組與動態數組
  3. 深入探索Java工作原理:JVM,內存回收及其他
責任編輯:金賀 來源: ITEYE
相關推薦

2011-06-16 10:02:08

JAVA靜態載入

2011-03-03 10:26:04

Pureftpd

2021-05-27 10:57:01

TCP定時器網絡協議

2009-06-25 13:43:00

Buffalo AJA

2010-10-21 16:24:18

sql server升

2010-03-16 15:23:32

java動態載入

2010-08-06 09:38:11

Flex讀取XML

2023-03-29 13:06:36

2024-09-13 08:27:00

2010-09-07 11:09:59

2010-07-27 15:03:37

Flex ArrayC

2010-05-10 18:19:00

負載平衡技術

2024-09-20 11:32:28

.NET內存管理

2010-07-14 10:30:26

Perl多線程

2011-03-23 11:22:14

oracle dbli

2010-07-15 14:38:55

Perl eval函數

2010-08-03 13:27:04

FlexBuilder

2016-11-07 09:02:02

Malloc內存syscall

2009-09-08 15:22:20

Spring依賴注入

2021-12-08 10:47:35

RabbitMQ 實現延遲
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美二区三区 | 国产精品久久久久久久7电影 | 国产玖玖 | 中文字幕视频在线免费 | 久久久久久久综合 | 久久精品国产久精国产 | 国产福利视频在线观看 | 亚洲综合一区二区三区 | 成人性视频免费网站 | 久久成人免费观看 | 欧美综合视频在线 | 日韩精品视频在线免费观看 | 成人免费视频在线观看 | 亚洲a级| 欧洲色| 午夜在线电影网 | 狠狠躁夜夜躁人人爽天天高潮 | 久久久久久久久久一区 | 午夜爽爽爽男女免费观看影院 | 日韩中文字幕 | 在线成人免费视频 | 97精品超碰一区二区三区 | 国产一区高清 | 久久国产综合 | 久久久久国产 | 欧美成人一区二区 | 久草资源 | 久在线观看 | 日本一道本 | 国产亚洲一区二区精品 | 午夜影院免费体验区 | 午夜欧美一区二区三区在线播放 | 欧美精品一区在线 | 久久r久久 | 国产精品mv在线观看 | 亚洲欧美一区二区在线观看 | 亚洲一区二区在线电影 | 久久69精品久久久久久久电影好 | 99国产精品一区二区三区 | 91福利网| 狠狠ri|