亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

探討java深拷貝

 更新時間:2016年02月14日 14:19:53   作者:楚巖  
這篇文章主要針對java深拷貝的相關(guān)內(nèi)容進行解析,幫助大家學習理解java深拷貝,感興趣的小伙伴們可以參考一下

本文將討論以下4個問題

    1. java Cloneable接口實現(xiàn)深拷貝
    2. java 序列化實現(xiàn)深拷貝
    3. 號稱最快的深拷貝二方庫cloning源碼分析
    4. 幾種拷貝方式速度的比較

深拷貝的概念本文就不說了。在C++中實現(xiàn)深拷貝一般情況下重載賦值操作符 “=” 來實現(xiàn)同一個類的對象間的深拷貝,所以很自然的在java中我們也同樣可以定義一個copy函數(shù),在函數(shù)內(nèi)部為對象的每一個屬性作賦值操作。這種方式簡單自然,但存在一個致命性的問題:如果有一天在類中新增加了一個需要深拷貝的屬性,那么相應的copy函數(shù)也得進行修改,這種方法給類的可擴展性帶來了極大的不方便。怎么解決這種問題,且看接下來的1、2、3章節(jié)的實現(xiàn)方式和4節(jié)的速度測試。
1. java Cloneable接口實現(xiàn)深拷貝
這種方式,需要類實現(xiàn)Colneable接口 clone 函數(shù),在clone函數(shù)中調(diào)用super.clone。這種方式的深拷貝同樣會帶來另一個問題,如果類中有其他類的對象作為屬性,則其他的類也需要重載并實現(xiàn)Cloneable接口。來一個例子,在下例中ComplexDO中包含了SimpleDO對象,要實現(xiàn)ComplexDO深拷貝,則需要先實現(xiàn)SimpleDO的clone接口:

public class SimpleDO implements Cloneable, Serializable {
    private int x = 1;
    private String s = "simpleDO";

    @Override
    protected Object clone() throws CloneNotSupportedException {
      SimpleDO newClass = (SimpleDO)super.clone();
      return newClass;
    }
  }

  public class ComplexDO implements Cloneable, Serializable {
    private int x = 1;
    private String s = "complex";
    private Integer a = 123;
    private Integer b = 1234;
    private Integer c = 1334455;
    private String s2 = "hehehe";
    private String s3 = "hahahaha";
    private Long id = 1233245L;
    private ArrayList<SimpleDO> l = new ArrayList<SimpleDO>();

    @Override
    public Object clone() throws CloneNotSupportedException {
      ComplexDO newClass = (ComplexDO) super.clone();
      newClass.l = new ArrayList<SimpleDO>();
      for (SimpleDO simple : this.l) {
        newClass.l.add((SimpleDO) simple.clone());
      }
      return newClass;
    }
  }

需要注意的是很多文章說String類型的對象賦值操作符是深拷貝,但是其實在java中使用賦值操作符的都屬于淺拷貝,但為什么這么明顯的錯誤這么多的文章會非要說這個是深拷貝呢?我的理解是String、類型的屬性都是基本類型,而且提供的方法只要是設(shè)計到內(nèi)部數(shù)據(jù)的變動都會new一個新的對象出來。所以一個String的操作不會影響到其原先指向的內(nèi)存。所以一般說String等基礎(chǔ)類的賦值操作為深拷貝。
由于這個原因,在使用String字符串拼接的時候,需要開辟新的內(nèi)存,所以很多人建議用StringBuilder來代替String來做拼接,因為StringBuilder只有在內(nèi)置的char數(shù)組范圍不夠的時候才重新申請更大的內(nèi)存(對于現(xiàn)代JVM,會對代碼調(diào)優(yōu),String+String會被優(yōu)化成StringBuilder.append的相類似的指令)。與拼接相對的裁剪,在String有個subString函數(shù),當使用subString函數(shù)時,新String的內(nèi)部char數(shù)組和原String是否相同?這個比較有意思,感興趣的可以對比看看JDK1.6和JKD1.7的實現(xiàn)。
2. java 序列化實現(xiàn)深拷貝
這種方式的原理是利用java序列化,將一個對象序列化成二進制字節(jié)流,然后對該字節(jié)流反序列化賦值給一個對象。代碼示例:

  public Object seirCopy(Object src) {
    try {
      ByteArrayOutputStream byteOut = new ByteArrayOutputStream();
      ObjectOutputStream out = new ObjectOutputStream(byteOut);
      out.writeObject(src);

      ByteArrayInputStream byteIn = new ByteArrayInputStream(byteOut.toByteArray());
      ObjectInputStream in = new ObjectInputStream(byteIn);
      Object dest = in.readObject();
      return dest;
    } catch (Exception e) {
      //do some error handler
      return null;
    }
 }

當然,也可以選用json等序列化的庫來完成序列化,這種方式有效的規(guī)避了Cloneabel接口的可擴展缺點,一個函數(shù)就可以基本上適用于所有的類.缺點是相對內(nèi)存拷貝,序列化需要先將對象轉(zhuǎn)換成二進制字節(jié)流,然后反序列化將該二進制字節(jié)流重新拷貝到一塊對象內(nèi)存,相對慢點。
3. 號稱最快的深拷貝二方庫cloning源碼分析
在源碼中,核心的處理邏輯在Cloner類中,
分兩條遞歸鏈路:

  • (1)deepClone->cloneInternal->fastClone->cloneInternal
  • (2)deepClone->cloneInternal->cloneObject->cloneInternal

在(1)中fastClone完成的是繼承自IfastCloner接口類的對象,即都是些集合操作的拷貝;
在(2)中cloneObject完成的是通過反射機制拿到普通對象的每一個屬性,然后對使用Objenesis新生成對象的屬性賦值。
這種方式可擴展性強,不僅可以依靠其現(xiàn)有的代碼完成深拷貝,還可以自己定義一些克隆的方式和不需要克隆的類型,靈活性強。
4. 幾種拷貝方式速度的比較
上述3中模式都可以完成深拷貝,那種拷貝的方式速度最快是我們所關(guān)心的。
先上測試代碼:
 

  public void testCloneComplex() throws CloneNotSupportedException {
    final int copyCount = 1;
    List<ComplexDO> complexDOList = new ArrayList<ComplexDO>(copyCount * 3);
    final ComplexDO complex = new ComplexDO();

    //調(diào)用二方庫
    long start = System.currentTimeMillis();
    for(int i = 0; i < copyCount; ++i) {
      final ComplexDO deepClone = cloner.deepClone(complex);
      complexDOList.add(deepClone);
    }
    long end = System.currentTimeMillis();
    System.out.println("deepClone cost time=" + (end-start));

    //調(diào)用Cloneable接口實現(xiàn)的clone函數(shù)
    start = System.currentTimeMillis();
    for(int i = 0; i < copyCount; ++i) {
      final ComplexDO interfaceClone = (ComplexDO) complex.clone();
      complexDOList.add(interfaceClone);
    }
    end = System.currentTimeMillis();
    System.out.println("interfaceClone cost time=" + (end-start));

    //序列化與反序列化生成新對象
    start = System.currentTimeMillis();
    for(int i = 0; i < copyCount; ++i) {
      final ComplexDO seirClone = seirCopy(complex);
      complexDOList.add(seirClone);
    }
    end = System.currentTimeMillis();
    System.out.println("seirClone cost time=" + (end-start));
  }


運行結(jié)果的單位為毫秒(此數(shù)據(jù)忽略不計算java熱點和可能的gc)。

從這個表可以得出結(jié)論:

1、實現(xiàn)Cloneable接口的拷貝是最快的,因為他只涉及到了內(nèi)存拷貝,但是如果涉及的屬性為普通對象比較多的時候?qū)懫饋砺闊c
2、序列化/反序列化拷貝最慢
3、使用cloning庫,由于使用了遞歸和反射機制相對Cloneable接口實現(xiàn)的拷貝要慢,但比序列化方式要快。

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助。

相關(guān)文章

  • java 判斷l(xiāng)ist是否為空過程解析

    java 判斷l(xiāng)ist是否為空過程解析

    這篇文章主要介紹了java 判斷l(xiāng)ist是否為空過程解析,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-09-09
  • Java C++實現(xiàn)相同MD5加密算法的方式

    Java C++實現(xiàn)相同MD5加密算法的方式

    這篇文章主要介紹了Java與C++實現(xiàn)相同MD5加密算法的方法,需要的朋友可以參考下面文章內(nèi)容
    2021-09-09
  • Java通過Fork/Join優(yōu)化并行計算

    Java通過Fork/Join優(yōu)化并行計算

    這篇文章主要為大家詳細介紹了Java通過Fork、Join來優(yōu)化并行計算,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-04-04
  • Java實現(xiàn)HashMap排序方法的示例詳解

    Java實現(xiàn)HashMap排序方法的示例詳解

    這篇文章主要通過一些示例為大家介紹了Java對HashMap進行排序的方法,幫助大家更好的理解和使用Java,感興趣的朋友可以了解一下
    2022-05-05
  • Java8中Optional類型和Kotlin中可空類型的使用對比

    Java8中Optional類型和Kotlin中可空類型的使用對比

    這篇文章主要給大家介紹了關(guān)于Java8中Optional類型和Kotlin中可空類型的使用對比,文中通過示例代碼給大家介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧。
    2017-09-09
  • Spring?Boot快速過濾出一次請求的所有日志

    Spring?Boot快速過濾出一次請求的所有日志

    這篇文章主要介紹了Spring?Boot快速過濾出一次請求的所有日志,本文講述了如何使用MDC工具來快速過濾一次請求的所有日志,并通過裝飾器模式使得MDC工具在異步線程里也能生效,需要的朋友可以參考下
    2022-11-11
  • springboot 傳參校驗@Valid及對其的異常捕獲方式

    springboot 傳參校驗@Valid及對其的異常捕獲方式

    這篇文章主要介紹了springboot 傳參校驗@Valid及對其的異常捕獲方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • Spring使用注解存儲Bean對象的方法詳解

    Spring使用注解存儲Bean對象的方法詳解

    在使用學習使用 Spring過程中,當我們要實現(xiàn)一個功能的時候,先應該考慮的是有沒有相應的注解是實現(xiàn)對應功能的,Spring 中很多功能的配置都是可以依靠注解實現(xiàn)的,而本篇中介紹的是使用注解來存儲 Bean 對象
    2023-07-07
  • 修改maven項目端口號的方法

    修改maven項目端口號的方法

    今天小編就為大家分享一篇修改maven項目端口號的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-05-05
  • RocketMQ-延遲消息的處理流程介紹

    RocketMQ-延遲消息的處理流程介紹

    這篇文章主要介紹了RocketMQ-延遲消息的處理流程,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-07-07

最新評論