中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java克隆對象的特性是什么

發布時間:2022-01-06 18:05:50 來源:億速云 閱讀:122 作者:iii 欄目:編程語言

本篇內容介紹了“Java克隆對象的特性是什么”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!

java面向對象的編程當中,要復制引用類型的對象,就必須克隆對象。通過調用對所有引用類型和對象都是可用的clone方法,來實現克隆。

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

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

public class Test {   public void someMethod() {   super.clone();   }   }

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

對象克隆

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

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

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

public class TestClone {   public static void main(String[] args) {   MyClone myClone1 = new MyClone("clone1");   MyClone myClone2 = (MyClone)myClone1.clone();   if (myClone2 != null) {   System.out.println(myClone2.getName());  System.out.println("myClone2 equals myClone1: " + myClone2.equals(myClone1));   } else {   System.out.println("Clone Not Supported");   }   } }   class MyClone {   private String name;   public MyClone(String name) {   this.name = name;   }  public String getName() {   return name;   }   public void setName(String name) {   this.name = name; }  public Object clone() {   try {  return super.clone();   } catch (CloneNotSupportedException e) {   return null;   }}

編譯執行TestClone,打印出:

C:\clone>javac *.java   C:\clone>java TestClone   Clone Not Supported   C:\clone>

說明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接口:

class MyClone implements Cloneable {   ...//其余不做改變   }   編譯執行TestClone,打印出:   C:\clone>javac *.java   C:\clone>java TestClone   clone1   myClone2 equals myClone1: false   C:\clone>

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

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

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

小結

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

public class A extends Cloneable {   public Object clone() {   try {   return super.clone();   } catch (CloneNotSupportedException e) {   //throw (new InternalError(e.getMessage()));   return null;   }   }   }

對象的深層次克隆

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

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

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

驗證對象的克隆方式

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

public class TestClone {   public static void main(String[] args) {   //為克隆對象設置值   MyClone myClone1 = new MyClone("clone1");   myClone1.setBoolValue(true);   myClone1.setIntValue(100);   //設置List值   List <Element>listValue = new ArrayList<Element>();   listValue.add(new Element("ListElement1"));   listValue.add(new Element("ListElement2"));   listValue.add(new Element("ListElement3"));   myClone1.setListValue(listValue);   //設置Element值   Element element1 = new Element("element1");   myClone1.setElement(element1);   //克隆   MyClone myClone2 = (MyClone)myClone1.clone();   if (myClone2 != null) {   //簡單屬性   System.out.println("myClone2.name=" + myClone2.getName()   + " myClone2.boolValue=" + myClone2.isBoolValue()   + " myClone2.intValue=" + myClone2.getIntValue() );   //復合屬性(List<Element>與Element)   List clonedList = myClone2.getListValue();   Element element2 = myClone2.getElement();   System.out.println("myClone2.listValue.size():" + clonedList.size());   System.out.println("myClone2.element.equals(myClone1.element):" + element2.equals(element1));   System.out.println("myClone2.element.name:" + element2.getName());  //下面我們測試一下myClone2.element是否等于myClone1.element   //以及myClone2.listValue是否等于myClone1.listValue   //為此,我們修改myClone2.element與myClone2.listValue,如果myClone1的相應值也跟著被修改了,則它們引用 的是同一個內存空間的變量,我們認為它們相等   clonedList.add("ListElement4");   System.out.println("myClone1.listValue.size():" + listValue.size());   element2.setName("Element2");   System.out.println("myClone1.element.name:" + element1.getName());   } else {   System.out.println("Clone Not Supported");   }   }   }   class MyClone implements Cloneable {   private int intValue;   private boolean boolValue;   private String name;   private List <Element>listValue;   private Element element;   public MyClone(String name) {   this.name = name;   }  ...//setter與getter方法(略)   }   class Element implements Cloneable {   private String name;   public Element (String name) {   this.name = name;   }   ...//setter與getter方法(略)   }

編譯執行TestClone,打印出:

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

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

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

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

class MyClone implements Cloneable {   ...   public Object clone() {   try {   MyClone myClone = (MyClone)super.clone();   //分別對屬性加以克隆操作   myClone.element = this.element.clone();   myClone.listValue = new ArrayList();   for (Element ele:this.listValue) {   myClone.listValue.add(ele.clone());   }   return myClone;   } catch (CloneNotSupportedException e) {   return null;   }   }   ...  }   //讓Element類也支持克隆操作   class Element implements Cloneable {   ...   public Element clone() {   try {   return (Element)super.clone();   } catch (CloneNotSupportedException e) {   return null;   }   }   }

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

“Java克隆對象的特性是什么”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

布拖县| 遂宁市| 丰台区| 阳新县| 灌南县| 德惠市| 卓尼县| 从江县| 林甸县| 西贡区| 嘉定区| 邛崃市| 翁牛特旗| 东方市| 神农架林区| 十堰市| 裕民县| 武乡县| 阿城市| 南城县| 兴安县| 桂阳县| 衡阳市| 无棣县| 会同县| 峡江县| 东至县| 溧水县| 玉门市| 宽甸| 珠海市| 东宁县| 家居| 沾益县| 普宁市| 瓦房店市| 英超| 大埔县| 长武县| 汉川市| 曲麻莱县|