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

簡述Java語言的對象克隆特性

開發 后端
本文通過介紹java.lang.Object#clone()方法來說明Java語言的對象克隆特性。java.lang.Object#clone()方法由java.lang.Object加以實現,主要對對象本身加以克隆。

Java中傳值及引伸深度克隆的思考中,我們講過引申到克隆技術Java中的所有對象都是Object類的子類。我們知道,Java是純面向對象的程序設計語言。Java里,所有的類的***父類都是java.lang.Object類,也就是說,如果一個類沒有顯示 申明繼承關系,它的父類默認就是java.lang.Object。

有一個很簡單的方法可以證明這一點,我們寫一個Test類,如下:

  1. public class Test {   
  2.     public void someMethod() {   
  3.         super.clone();   
  4.     }   
  5. }   
  6.  

里面調用了super.clone(),編譯時并不報錯。其實clone()方法為java.lang.Object類提供的一個 protected型方法。

對象克隆

本文通過介紹java.lang.Object#clone()方法來說明Java語言的對象克隆特性。

java.lang.Object#clone()方法由java.lang.Object加以實現,主要對對象本身加以克隆。

首先我們看看下面的例子:

  1. public class TestClone {   
  2.     public static void main(String[] args) {   
  3.         MyClone myClone1 = new MyClone("clone1");   
  4.           
  5.         MyClone myClone2 = (MyClone)myClone1.clone();   
  6.           
  7.         if (myClone2 != null) {   
  8.             System.out.println(myClone2.getName());   
  9.             System.out.println("myClone2 equals myClone1: " + myClone2.equals(myClone1));   
  10.         } else {   
  11.             System.out.println("Clone Not Supported");   
  12.         }   
  13.     }   
  14. }   
  15. class MyClone {   
  16.     private String name;   
  17.     public MyClone(String name) {   
  18.         this.name = name;   
  19.     }   
  20.       
  21.     public String getName() {   
  22.         return name;   
  23.     }   
  24.     public void setName(String name) {   
  25.         this.name = name;   
  26.     }   
  27.       
  28.     public Object clone() {   
  29.         try {   
  30.             return super.clone();   
  31.         } catch (CloneNotSupportedException e) {   
  32.             return null;   
  33.         }   
  34.     }   
  35.  

編譯執行TestClone,打印出:

  1. C:\clone>javac *.java   
  2. C:\clone>java TestClone   
  3. Clone Not Supported   
  4. C:\clone>   
  5.  

說明MyClone#clone()方法調用super.clone()時拋出了CloneNotSupportedException異常,不支持克隆。

為什么父類java.lang.Object里提供了clone()方法,卻不能調用呢?

原來,Java語言雖然提供了這個方法,但考慮到安全問題, 一方面將clone()訪問級別設置為protected型,以限制外部類訪問;

另一方面,強制需要提供clone功能的子類實現java.lang.Cloneable接口,在運行期,JVM會檢查調用clone()方法的 類,如果該類未實現java.lang.Cloneable接口,則拋出CloneNotSupportedException異常。

java.lang.Cloneable接口是一個空的接口,沒有申明任何屬性與方法。該接口只是告訴JVM,該接口的實現類需要開放“克隆”功能。

我們再將MyClone類稍作改變,讓其實現Cloneable接口:

  1. class MyClone implements Cloneable {   
  2.     ...//其余不做改變   
  3. }   
  4.  
  5. 編譯執行TestClone,打印出:   
  6.  
  7. C:\clone>javac *.java   
  8. C:\clone>java TestClone   
  9. clone1   
  10. myClone2 equals myClone1: false   
  11. C:\clone>   
  12.  

根據結果,我們可以發現:

1,myClone1.clone()克隆了跟myClone1具有相同屬性值的對象

2,但克隆出的對象myClone2跟myClone1不是同一個對象(具有不同的內存空間)

小結

如果要讓一個類A提供克隆功能,該類必須實現java.lang.Cloneable接口,并重載 java.lang.Object#clone()方法。

  1. public class A extends Cloneable {   
  2.     public Object clone() {   
  3.         try {   
  4.             return super.clone();   
  5.         } catch (CloneNotSupportedException e) {   
  6.             //throw (new InternalError(e.getMessage()));   
  7.             return null;   
  8.         }   
  9.     }   
  10. }   
  11.  

對象的深層次克隆

上例說明了怎么樣克隆一個具有簡單屬性(String,int,boolean等)的對象。

但如果一個對象的屬性類型是List,Map,或者用戶自定義的其他類時,克隆行為是通過怎樣的方式進行的?

很多時候,我們希望即使修改了克隆后的對象的屬性值,也不會影響到原對象,這種克隆我們稱之為對象的深層次克隆。怎么樣實現對象的深層次克隆呢?

驗證對象的克隆方式

為了驗證對象的克隆方式,我們對上面的例子加以改進,如下(為了節省篇幅,我們省略了setter與getter方法):

  1. public class TestClone {   
  2.     public static void main(String[] args) {   
  3.         //為克隆對象設置值   
  4.          MyClone myClone1 = new MyClone("clone1");   
  5.          myClone1.setBoolValue(true);   
  6.          myClone1.setIntValue(100);   
  7.           
  8.         //設置List值   
  9.          List <Element>listValue = new ArrayList<Element>();   
  10.          listValue.add(new Element("ListElement1"));   
  11.          listValue.add(new Element("ListElement2"));   
  12.          listValue.add(new Element("ListElement3"));   
  13.          myClone1.setListValue(listValue);   
  14.           
  15.         //設置Element值   
  16.          Element element1 = new Element("element1");   
  17.          myClone1.setElement(element1);   
  18.           
  19.           
  20.         //克隆   
  21.          MyClone myClone2 = (MyClone)myClone1.clone();   
  22.           
  23.         if (myClone2 != null) {   
  24.               
  25.             //簡單屬性   
  26.              System.out.println("myClone2.name=" + myClone2.getName()   
  27.                      + " myClone2.boolValue=" + myClone2.isBoolValue()   
  28.                      + " myClone2.intValue=" + myClone2.getIntValue() );   
  29.               
  30.             //復合屬性(List<Element>與Element)   
  31.              List clonedList = myClone2.getListValue();   
  32.              Element element2 = myClone2.getElement();   
  33.               
  34.              System.out.println("myClone2.listValue.size():" + clonedList.size());   
  35.              System.out.println("myClone2.element.equals(myClone1.element):" + element2.equals(element1));   
  36.              System.out.println("myClone2.element.name:" + element2.getName());   
  37.               
  38.             //下面我們測試一下myClone2.element是否等于myClone1.element   
  39.             //以及myClone2.listValue是否等于myClone1.listValue   
  40.             //為此,我們修改myClone2.element與myClone2.listValue,如果myClone1的相應值也跟著被修改了,則它們引用 的是同一個內存空間的變量,我們認為它們相等   
  41.               
  42.              clonedList.add("ListElement4");   
  43.               
  44.              System.out.println("myClone1.listValue.size():" + listValue.size());   
  45.               
  46.              element2.setName("Element2");   
  47.              System.out.println("myClone1.element.name:" + element1.getName());   
  48.               
  49.          } else {   
  50.              System.out.println("Clone Not Supported");   
  51.          }          
  52.           
  53.      }   
  54.  
  55. }   
  56.  
  57.  
  58. class MyClone implements Cloneable {   
  59.     private int intValue;   
  60.     private boolean boolValue;   
  61.     private String name;   
  62.     private List <Element>listValue;   
  63.     private Element element;   
  64.  
  65.     public MyClone(String name) {   
  66.          this.name = name;   
  67.      }   
  68.  
  69.      ...//setter與getter方法(略)   
  70. }   
  71.  
  72. class Element implements Cloneable   {   
  73.     private String name;   
  74.       
  75.     public Element (String name) {   
  76.          this.name = name;   
  77.      }   
  78.  
  79.      ...//setter與getter方法(略)   
  80. }   
  81.  

編譯執行TestClone,打印出:

  1. C:\clone>javac *.java   
  2. C:\clone>java TestClone   
  3. myClone2.name=clone1 myClone2.boolValue=true myClone2.intValue=100   
  4. myClone2.listValue.size():3   
  5. myClone2.element.equals(myClone1.element):true   
  6. myClone2.element.name:element1   
  7. myClone1.listValue.size():4   
  8. myClone1.element.name:Element2   
  9. myClone2 equals myClone1: false   
  10. C:\clone>   
  11.  

我們發現,對于對象里的List,Element等復合屬性,super.clone()只是簡單地賦值,沒有采取克隆手段。也就是說,修改被克 隆后的對象值,會影響到原對象。

怎么進行深層次的克隆呢?

答案是,我們只能手動在重載的clone()方法里,對屬性也分別采用克隆操作。當然條件是,屬性類也得支持克隆操作

  1. class MyClone implements Cloneable {   
  2.      ...   
  3.     public Object clone() {   
  4.         try {   
  5.              MyClone myClone = (MyClone)super.clone();   
  6.             //分別對屬性加以克隆操作   
  7.              myClone.element = this.element.clone();   
  8.               
  9.              myClone.listValue = new ArrayList();   
  10.             for (Element ele:this.listValue) {   
  11.                  myClone.listValue.add(ele.clone());   
  12.              }   
  13.                           
  14.             return myClone;   
  15.          } catch (CloneNotSupportedException e) {   
  16.             return null;   
  17.          }   
  18.      }   
  19.      ...   
  20. }   
  21.  
  22. //讓Element類也支持克隆操作   
  23. class Element implements Cloneable   {   
  24.      ...   
  25.     public Element clone() {   
  26.         try {   
  27.             return (Element)super.clone();   
  28.          } catch (CloneNotSupportedException e) {   
  29.             return null;   
  30.          }   
  31.      }   
  32. }   
  33.  

深層次的克隆操作往往存在效率問題,尤其是需要讓List,Map等集合類也支持深層次的克隆操作時。

總結

本文結合范例,比較深入地介紹了Java語言的克隆屬性,以及克隆的實現方法等。同時分析了深層次克隆的概念,實現,以及存在的問題等。 但是有沒有更好的方法呢?當然,是有的,串行化來實現。

【編輯推薦】

  1. Java中傳值及引伸深度克隆的思考
  2. 淺析Java中的深拷貝與淺拷貝
  3. 對Java本身API庫的擴充:Jython庫列表
  4. JAVA 對象拷貝
  5. Java新手進階:細說引用類型
責任編輯:佚名 來源: 博客園
相關推薦

2011-06-28 11:27:19

java克隆對象

2018-04-03 13:10:27

Java對象克隆

2011-05-10 10:00:41

克隆

2012-03-13 16:39:52

Java

2009-05-21 09:08:52

接口C++面向對象

2013-03-26 10:27:01

JavaScriptjson

2022-08-16 14:27:56

Java開發編程

2021-06-03 10:00:47

JavaScript 前端數克隆對象

2013-12-12 16:37:12

Lua腳本語言

2010-01-15 16:59:52

VB.NET對象克隆

2021-07-15 23:18:48

Go語言并發

2011-12-01 11:24:13

2010-03-08 15:12:27

Python語言

2009-09-24 10:15:37

Hibernate查詢

2009-08-26 09:26:12

C#語言層次劃分

2010-07-12 10:02:26

HART協議

2010-01-22 13:28:13

VB.NET對象克隆

2010-01-06 15:35:06

JSON對象

2009-09-24 17:28:29

Hibernate S

2009-09-18 09:59:39

C# CLR
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久在线视频 | 一区二区国产精品 | 久久久.com| 51ⅴ精品国产91久久久久久 | 黄色片网站在线观看 | 一级在线免费观看 | 色播视频在线观看 | 亚洲欧美综合精品另类天天更新 | 欧美成人自拍 | 国家aaa的一级看片 h片在线看 | 成人久久18免费网站麻豆 | 岛国av一区二区 | 天堂一区二区三区四区 | 久一精品 | 久草在线在线精品观看 | www.日日操 | 久久视频免费看 | 午夜电影网 | 日韩精品专区在线影院重磅 | av一区二区三区四区 | 日韩在线不卡 | 亚洲一级毛片 | 成人不卡| 日韩欧美专区 | 操操日| 亚洲成人免费在线 | 91精品国产色综合久久 | а√中文在线8 | 我爱操| 亚洲成人一区二区 | 国产免费人成xvideos视频 | 罗宾被扒开腿做同人网站 | 国产精品色婷婷久久58 | 精品国产一区二区三区av片 | 国产成人午夜高潮毛片 | 97av视频| 成人三级视频在线观看 | 9porny九色视频自拍 | 亚洲第一中文字幕 | 成人黄色av网址 | 91麻豆精品国产91久久久更新资源速度超快 |