欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解Java中clone的寫(xiě)法

 更新時(shí)間:2018年07月27日 08:30:46   作者:mozi_song  
這篇文章主要介紹了Java中clone的寫(xiě)法,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下

Cloneable這個(gè)接口設(shè)計(jì)得十分奇葩,不符合正常人的使用習(xí)慣,然而用這個(gè)接口的人很多也很有必要,所以還是有必要了解一下這套扭曲的機(jī)制。以下內(nèi)容來(lái)自于對(duì)Effective Java ed 2. item 11的整理。

 Cloneable接口

首先,Cloneable接口中并沒(méi)有方法。它的存在意義一是讓程序員注明當(dāng)前對(duì)象可以clone,二是改變父類(lèi)Object類(lèi)中clone方法的行為:如果某個(gè)類(lèi)實(shí)現(xiàn)了Cloneable,那么它的父類(lèi)Object的clone方法可以調(diào)用,否則會(huì)拋出CloneNotSupportedException。(奇葩吧)

也就是說(shuō),如果我們要告訴用戶(hù),這個(gè)類(lèi)是可以clone的,并且在我們的實(shí)現(xiàn)中需要調(diào)用super.clone,那么我們就必須實(shí)現(xiàn)Cloneable。

(然而,即使某個(gè)類(lèi)實(shí)現(xiàn)了Cloneable,也不一定保證它就有clone方法,這是這個(gè)接口設(shè)計(jì)的奇葩之處之一,設(shè)計(jì)者可能是反社會(huì)吧) 

我們的clone方法

需要重寫(xiě)clone方法的情況分為兩類(lèi)。

    1:需要實(shí)現(xiàn)Cloneable接口。

    2:只需要重寫(xiě)clone方法。

其中,第一種情況比較普遍。第二種可以看作為了討論的完整性對(duì)第一種進(jìn)行的補(bǔ)充。

需要實(shí)現(xiàn)Cloneable接口

考慮到clone方法是直接給用戶(hù)用的,建議做到以下幾點(diǎn):

將限制符改為public;

將它的返回類(lèi)型設(shè)置成子類(lèi)類(lèi)型(可以這么做是因?yàn)閖ava允許covariant return type);

接住CloneNotSupportedException并不再拋出(既然已經(jīng)實(shí)現(xiàn)了Cloneable接口,就不會(huì)拋出這個(gè)異常,不然用戶(hù)又要在

那里try-catch半天)。

@Override
public PhoneNumber clone() throws ... {
  try {
   return (PhoneNumber) super.clone();
  } catch(CloneNotSupportedException e) {
   throw new AssertionError(); // Can't happen
  }
}

注意,這里給出的是clone方法的大體寫(xiě)法,包括函數(shù)簽名等,先讓你有一個(gè)大略的方向。當(dāng)我們按照以上三條搭好clone方法的框框后,具體如何去實(shí)現(xiàn)克隆的過(guò)程,下一節(jié)會(huì)舉例詳述。

注:如果當(dāng)前類(lèi)是final的,可以直接使用構(gòu)造器來(lái)構(gòu)造對(duì)象。(如果不是final的,那么可能還會(huì)有子類(lèi),子類(lèi)再調(diào)用super.clone的時(shí)候就只能返回父類(lèi)類(lèi)型對(duì)象,就不太合適了,所以只有final類(lèi)適合用構(gòu)造器)

只需要重寫(xiě)clone方法

這個(gè)類(lèi)可能是繼承鏈上的一個(gè)中間類(lèi)。此時(shí)該clone方法最好模擬Object.clone的行為,即:

限制符為protected;

不實(shí)現(xiàn)Cloneable;

拋出CloneNotSupportedException。

不同情景下的clone方法實(shí)現(xiàn)

首先,應(yīng)熟悉Object.clone的行為(因?yàn)樵谖覀冏约旱念?lèi)中經(jīng)常會(huì)調(diào)用super.clone,最終調(diào)用Object.clone):淺拷貝。即:先創(chuàng)建一個(gè)新對(duì)象,然后將它的所有域初始化為待拷貝對(duì)象的域的對(duì)應(yīng)值。

另外,所有數(shù)組都會(huì)實(shí)現(xiàn)Cloneable接口,T[].clone的返回類(lèi)型也為T(mén)[],行為與Object類(lèi)似。(這是一個(gè)好用的feature,實(shí)現(xiàn)淺拷貝時(shí)會(huì)經(jīng)常用到)

官方文檔對(duì)clone的實(shí)現(xiàn)建議是:先調(diào)用super.clone創(chuàng)建對(duì)象;如果對(duì)象的域都是基本類(lèi)型,則一切搞定;否則,如果對(duì)象是可變對(duì)象,則要將組成對(duì)象的"deep structure"的對(duì)象全部復(fù)制,然后將復(fù)制品的域引用指向這些復(fù)制后的對(duì)象。 

上一節(jié)給出的PhoneNumber的clone屬于前者(對(duì)象域?yàn)殡娫捥?hào)碼、區(qū)號(hào)等,為基本類(lèi)型short),所以調(diào)用super.clone再加一個(gè)cast就可以搞定。

注意這個(gè)藍(lán)色的deep structure,指明了clone方法實(shí)現(xiàn)的精髓。以下舉兩個(gè)例子,讀者可細(xì)細(xì)品味。

案例一:Stack

public class Stack {
 private Object[] elements;
 private int size = 0;
 private static final int DEFAULT_INITIAL_CAPACITY = 16;
 public Stack() {...}
 public void push(Object e) {...}
 public Object pop() {...}
 private void ensureCapacity() {...} //omitted for simplicity
}

如果在Stack的clone方法中,也簡(jiǎn)單地返回super.clone,會(huì)有一個(gè)嚴(yán)重的后果,就是在原對(duì)象中如果增刪了元素,在復(fù)制對(duì)象中的size不變,但是實(shí)際上元素被增刪了,違反了復(fù)制對(duì)象的invariant。

解決辦法是將elements數(shù)組獨(dú)立克隆:

@Override public Stack clone() {
 try {
  Stack result = (Stack) super.clone();
  result.elements = elements.clone();
  return result;
 } catch (CloneNotSupportedException e) {
  throw new AssertionError();
 }
}

兩種方法的區(qū)別如下:(渣圖……)

第一種方法對(duì)應(yīng)左圖,由于克隆后的對(duì)象的elements指向原對(duì)象中的數(shù)組,當(dāng)原對(duì)象增刪元素時(shí),克隆后的對(duì)象的backing array也跟著自動(dòng)變化。第二種方法對(duì)應(yīng)右圖,克隆后對(duì)象的數(shù)組和原對(duì)象的數(shù)組是互相獨(dú)立的,當(dāng)原對(duì)象增刪元素時(shí),克隆后的對(duì)象可以不受影響,因?yàn)樗€保持原有的那些引用。雖然兩種都是淺拷貝,但只有第二種符合不變性。而且第二種是容器類(lèi)的一種常用做法,如ArrayList的copy constructor。

案例二:HashTable

在Stack的基礎(chǔ)上再?gòu)?fù)雜一點(diǎn),我們研究一個(gè)HashTable:

public class HashTable implements Cloneable {
  private Entry[] buckets = ...;
  private static class Entry {
   final Object key;
   Object value;
   Entry next;
   Entry(Object key, Object value, Entry next) {
     this.key = key;
     this.value = value;
     this.next = next;
   }
  }
  ... // Remainder omitted
}

如果我們照搬Stack的克隆方法,是否會(huì)有效呢?

@Override public HashTable clone() {
  try {
   HashTable result = (HashTable) super.clone();
   result.buckets = buckets.clone();
   return result;
  } catch (CloneNotSupportedException e) {
   throw new AssertionError();
  }
}

克隆后的HashTable有自己的array了,看起來(lái)好像沒(méi)什么問(wèn)題了。然而,HashTable使用的是Entry對(duì)象頭尾相接的鏈表??寺『驟ntry元素們還指向同樣的對(duì)象,此時(shí)如果原table增刪了元素,其實(shí)質(zhì)是它將某些Entry指向了新Entry或指向null;由于克隆后的table與克隆前的table共享一套Entry對(duì)象,所以它的內(nèi)部結(jié)構(gòu)發(fā)生了同樣的改變,但它并不知道自己發(fā)生了改變,這樣就出現(xiàn)了奇怪的現(xiàn)象,比如說(shuō)克隆后的table的size明明沒(méi)變,卻憑空多出/消失了一些元素。

HashTable original = new HashTable();
original.put(x, y);
HashTable cloned = original.clone();
original.remove(x); //cloned gets removed by one element too, but does not know of it!!
if(cloned.size() > 0){
  doSomething(); //Danger! It's actually empty!!  
}

如圖:

解決方法是將其中value的容器Entry做深拷貝。

public class HashTable implements Cloneable {
  private Entry[] buckets = ...;
  private static class Entry {
   final Object key;
   Object value;
   Entry next;
   Entry(Object key, Object value, Entry next) {
   this.key = key;
   this.value = value;
   this.next = next;
   // Recursively copy the linked list headed by this Entry
   Entry deepCopy() {
     return new Entry(key, value, next == null ? null : next.deepCopy());
   }  
 }
  @Override public HashTable clone() {
   try {
     HashTable result = (HashTable) super.clone();
     result.buckets = new Entry[buckets.length];
     for (int i = 0; i < buckets.length; i++)
      if (buckets[i] != null)
       result.buckets[i] = buckets[i].deepCopy();
     return result;
   } catch (CloneNotSupportedException e) {
     throw new AssertionError();
   }
  }
  ... // Remainder omitted
}


注:value指向的Object仍然沒(méi)變,所以這種方法只是在一定程度上做深拷貝。由于HashTable直接操作的是Entry,將Entry這一層深拷貝即可。

由于上述deepCopy()方法容易引起stack overflow,作者建議使用iteration代替recursion.

//Iteratively copy the linked list headed by this Entry
Entry deepCopy() {
  Entry result = new Entry(key, value, next);
  for (Entry p = result; p.next != null; p = p.next)
   p.next = new Entry(p.next.key, p.next.value, p.next.next);
  return result;
}

其他碎碎念

(非final類(lèi)的)clone方法不應(yīng)調(diào)用克隆后對(duì)象的nonfinal方法。若該類(lèi)的子類(lèi)重寫(xiě)了這個(gè)nonfinal方法,該方法有可能在子類(lèi)創(chuàng)建完畢之前去調(diào)用它的一些方法/數(shù)據(jù),可能會(huì)引起數(shù)據(jù)損壞。

如果類(lèi)中有一個(gè)指向可變對(duì)象的final域,則以上的clone實(shí)現(xiàn)機(jī)制無(wú)法work,因?yàn)閷?duì)象創(chuàng)建好以后無(wú)法再給final域assign一個(gè)值。

不可變類(lèi)不應(yīng)該支持clone,因?yàn)閏lone后的對(duì)象跟原對(duì)象沒(méi)有區(qū)別。
其實(shí)一種比較好的方法是copy constructor或copy factory。它們沒(méi)有Cloneable的那些奇葩性,不拋異常,而且可以搞定final域。

public Yum(Yum yum); //copy constructor
public static Yum newInstance(Yum yum); //copy factory

一個(gè)更好的好處是,interface-based copy constructor或copy factory (稱(chēng)為conversion constructors / conversion factories)可以允許用戶(hù)選擇與原對(duì)象不同類(lèi)的克隆對(duì)象。如

HashSet s = ...;
new TreeSet(s); //將HashSet轉(zhuǎn)換成TreeSet

總結(jié)

 以上所述是小編給大家介紹的Java中clone的寫(xiě)法,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!

相關(guān)文章

  • SpringBoot集成nacos動(dòng)態(tài)刷新數(shù)據(jù)源的實(shí)現(xiàn)示例

    SpringBoot集成nacos動(dòng)態(tài)刷新數(shù)據(jù)源的實(shí)現(xiàn)示例

    這篇文章主要介紹了SpringBoot集成nacos動(dòng)態(tài)刷新數(shù)據(jù)源的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-12-12
  • Java自定義異常類(lèi)的實(shí)例詳解

    Java自定義異常類(lèi)的實(shí)例詳解

    這篇文章主要介紹了Java自定義異常類(lèi)的實(shí)例詳解的相關(guān)資料,希望通過(guò)本文能幫助到大家,讓大家學(xué)習(xí)理解掌握這部分內(nèi)容,需要的朋友可以參考下
    2017-09-09
  • Spring Boot整合RabbitMQ實(shí)例(Topic模式)

    Spring Boot整合RabbitMQ實(shí)例(Topic模式)

    Topic Exchange 轉(zhuǎn)發(fā)消息主要是根據(jù)通配符。接下來(lái)通過(guò)本文給大家分享Spring Boot整合RabbitMQ實(shí)例(Topic模式),需要的朋友參考下吧
    2017-04-04
  • Mybatis Order by動(dòng)態(tài)參數(shù)防注入方式

    Mybatis Order by動(dòng)態(tài)參數(shù)防注入方式

    這篇文章主要介紹了Mybatis Order by動(dòng)態(tài)參數(shù)防注入方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-04-04
  • java.lang.ClassCastException的問(wèn)題解決

    java.lang.ClassCastException的問(wèn)題解決

    本文主要介紹了java.lang.ClassCastException的問(wèn)題解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2024-06-06
  • Java之判斷2000~2023年有哪些年份是閏年并打印輸出

    Java之判斷2000~2023年有哪些年份是閏年并打印輸出

    這篇文章主要介紹了Java之判斷2000~2023年有哪些年份是閏年并打印輸出,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-12-12
  • Java數(shù)組使用binarySearch()方法查找指定元素的實(shí)現(xiàn)

    Java數(shù)組使用binarySearch()方法查找指定元素的實(shí)現(xiàn)

    這篇文章主要介紹了Java數(shù)組使用binarySearch()方法查找指定元素的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-01-01
  • Java中串行接口調(diào)用優(yōu)化方式

    Java中串行接口調(diào)用優(yōu)化方式

    這篇文章主要介紹了Java中串行接口調(diào)用優(yōu)化方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-05-05
  • 一文帶你快速了解java中的static關(guān)鍵詞

    一文帶你快速了解java中的static關(guān)鍵詞

    這篇文章主要給大家介紹了關(guān)于java中static關(guān)鍵詞的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-12-12
  • Spring擴(kuò)展點(diǎn)之BeanFactoryPostProcessor詳解

    Spring擴(kuò)展點(diǎn)之BeanFactoryPostProcessor詳解

    這篇文章主要介紹了Spring擴(kuò)展點(diǎn)之BeanFactoryPostProcessor詳解,Spring的設(shè)計(jì)非常優(yōu)雅,有很多的擴(kuò)展點(diǎn)供我們對(duì)項(xiàng)目進(jìn)行擴(kuò)展,今天學(xué)習(xí)一下Spring其中擴(kuò)展點(diǎn)之一的BeanFactoryPostProcessor,需要的朋友可以參考下
    2023-11-11

最新評(píng)論