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

詳解java中的深拷貝和淺拷貝(clone()方法的重寫、使用序列化實現(xiàn)真正的深拷貝)

 更新時間:2019年03月29日 15:13:23   作者:changshuchao  
這篇文章主要介紹了java中的深拷貝和淺拷貝(clone()方法的重寫、使用序列化實現(xiàn)真正的深拷貝),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

1.序列化實現(xiàn)

public class CloneUtils {

  @SuppressWarnings("unchecked")
  public static <T extends Serializable> T clone(T object){

    T cloneObj = null;
    try {
      ByteArrayOutputStream out = new ByteArrayOutputStream();
      ObjectOutputStream obs = new ObjectOutputStream(out);
      obs.writeObject(object);
      obs.close();

      ByteArrayInputStream ios = new ByteArrayInputStream(out.toByteArray());
      ObjectInputStream ois = new ObjectInputStream(ios);
      cloneObj = (T) ois.readObject();



    }catch (Exception e){
      e.printStackTrace();
    }
    return cloneObj;

  }

}

2.主代碼

public class TestString {
  public static void main(String[] args) {
    TestString test = new TestString();
    System.out.println("-------淺拷貝---------");
    test.qianCopyTest();

    System.out.println();

    System.out.println("--------使用clone深拷貝--------");
    test.defaultCloneTest();

    System.out.println();

    System.out.println("--------使用序列化實現(xiàn)對象的拷貝--------");
    test.streamClonrTest();


    System.out.println("--------耗時對比--------");
    System.out.println("耗時1 : "+ test.qianCopyCost());
    System.out.println("耗時2 : "+ test.CloneCopyCost());
    System.out.println("耗時3 : "+ test.StreamCopyCost());
  }

  /*淺拷貝*/
  private void qianCopyTest() {
    String s = "cd";
    change(s);
    System.out.println(s);
    System.out.println("----------------");
    String b = new String("cd");
    change(b);
    System.out.println(b);
    System.out.println("----------------");
    int me = 1;
    change(me);
    System.out.println(me);
    System.out.println("----------------");
    Person person = new Person("我", 13,new Email("我"));
    change(person);
    System.out.println(person.toString());
  }

  /*使用默認(rèn)的clone方法,需要Person實現(xiàn)Cloneable接口*/
  private void defaultCloneTest(){
    Person person = new Person("我", 13,new Email("我"));
    Person person1 = person.clone();
    Person person2 = person.clone();
    System.out.println("person : 【"+person+"】");
    System.out.println("person1 : 【"+person1+"】");
    System.out.println("person2 : 【"+person2+"】");
    //改一個就會觸動全部??! 這就是使用默認(rèn)的clone方法的弊端
    /*該clone()方法是使用Object類的clone()方法,但是該方法存在一個缺陷,它并不會將對象的所有屬性全部拷貝過來,而是有選擇性的拷貝,基本規(guī)則如下:
       1、 基本類型
         如果變量是基本很類型,則拷貝其值,比如int、float等。
       2、 對象
         如果變量是一個實例對象,則拷貝其地址引用,也就是說此時新對象與原來對象是公用該實例變量。
       3、 String字符串
         若變量為String字符串,則拷貝其地址引用。但是在修改時,它會從字符串池中重新生成一個新的字符串,原有紫都城對象保持不變。*/
    person.getEmail().setContent("你");
    System.out.println("之后的person : 【"+person+"】");
    System.out.println("之后的person1 : 【"+person1+"】");
    System.out.println("之后的person2 : 【"+person2+"】");
  }

  /*使用序列化實現(xiàn)對象的拷貝,需要對象以及對象中的其他對象都要實現(xiàn)Serializable接口*/
  private void streamClonrTest(){
    Person person = new Person("我", 13,new Email("我"));
    Person person1 = CloneUtils.clone(person);
    Person person2 = CloneUtils.clone(person);
    System.out.println("person : 【"+person+"】");
    System.out.println("person1 : 【"+person1+"】");
    System.out.println("person2 : 【"+person2+"】");
    person.getEmail().setContent("你");
    System.out.println("之后的person : 【"+person+"】");
    System.out.println("之后的person1 : 【"+person1+"】");
    System.out.println("之后的person2 : 【"+person2+"】");
  }

  private static void change(String x) {
    x = "ab";
  }

  private static void change(int x) {
    x = 2;
  }

  private static void change(Person x) {
    x = new Person("你", 20, new Email("你"));
  }

  private long qianCopyCost(){
    long start = System.currentTimeMillis();
    Person person = new Person("我", 13,new Email("我"));
    List<Person> list = new ArrayList<>();
    for(int i = 0;i<=10000;i++){
      list.add(new Person("你", 20, new Email("你")));
    }
    return System.currentTimeMillis()-start;
  }

  private long CloneCopyCost(){
    long start = System.currentTimeMillis();
    Person person = new Person("我", 13,new Email("我"));
    List<Person> list = new ArrayList<>();
    for(int i = 0;i<=10000;i++){
      list.add(person.clone());
    }
    return System.currentTimeMillis()-start;
  }

  private long StreamCopyCost(){
    long start = System.currentTimeMillis();
    Person person = new Person("我", 13,new Email("我"));
    List<Person> list = new ArrayList<>();
    for(int i = 0;i<=10000;i++){
      list.add(CloneUtils.clone(person));
    }
    return System.currentTimeMillis()-start;
  }

}

class Person implements Serializable, Cloneable {

  private static final long serialVersionUID = -8584225043397465132L;
  private String name;
  private int age;

  public void setEmail(Email email) {
    this.email = email;
  }

  private Email email;

  public Email getEmail() {
    return email;
  }

  public void setName(String name) {
    this.name = name;
  }

  public void setAge(int age) {
    this.age = age;
  }

  public Person(String name, int age, Email email) {
    this.name = name;
    this.age = age;
    this.email = email;
  }

  @Override
  public String toString() {
    return "name : " + name + " | age : " + age +" | content : "+email.getContent();
  }

  @Override
  protected Person clone() {
    Person person = null;
    try {
      person = (Person) super.clone();
      /*如果加上下一行 “使用clone深拷貝” 就不會改一處其他都改變了*/
      person.setEmail(new Email(person.getEmail().getContent()));
    } catch (CloneNotSupportedException e) {
      e.printStackTrace();
    }

    return person;
  }
}

class Email implements Serializable {

  private static final long serialVersionUID = 1426052929769365539L;
  private String content;

  public void setContent(String content) {
    this.content = content;
  }

  public String getContent() {
    return content;
  }

  public Email(String content) {
    this.content = content;
  }
}

測試了一下時間:
輸出:
-------淺拷貝---------
cd

cd

1

name : 我 | age : 13 | content : 我

--------使用clone深拷貝--------
person : 【name : 我 | age : 13 | content : 我】
person1 : 【name : 我 | age : 13 | content : 我】
person2 : 【name : 我 | age : 13 | content : 我】
之后的person : 【name : 我 | age : 13 | content : 你】
之后的person1 : 【name : 我 | age : 13 | content : 我】
之后的person2 : 【name : 我 | age : 13 | content : 我】

--------使用序列化實現(xiàn)對象的拷貝--------
person : 【name : 我 | age : 13 | content : 我】
person1 : 【name : 我 | age : 13 | content : 我】
person2 : 【name : 我 | age : 13 | content : 我】
之后的person : 【name : 我 | age : 13 | content : 你】
之后的person1 : 【name : 我 | age : 13 | content : 我】
之后的person2 : 【name : 我 | age : 13 | content : 我】
--------耗時對比--------
耗時1 : 2
耗時2 : 1
耗時3 : 338

以上所述是小編給大家介紹的java中的深拷貝和淺拷貝(clone()方法的重寫、使用序列化實現(xiàn)真正的深拷貝)詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!

相關(guān)文章

  • Java面試題及答案集錦(基礎(chǔ)題122道,代碼題19道)

    Java面試題及答案集錦(基礎(chǔ)題122道,代碼題19道)

    本文是小編收集整理的關(guān)于java基礎(chǔ)面試題及答案集錦,基礎(chǔ)題目有122道,代碼題目有19道,非常不錯,值得收藏,需要的朋友參考下
    2017-01-01
  • Java面試題沖刺第二十六天--實戰(zhàn)編程2

    Java面試題沖刺第二十六天--實戰(zhàn)編程2

    這篇文章主要為大家分享了最有價值的三道java實戰(zhàn)編程的面試題,涵蓋內(nèi)容全面,包括數(shù)據(jù)結(jié)構(gòu)和算法相關(guān)的題目、經(jīng)典面試編程題等,感興趣的小伙伴們可以參考一下
    2021-08-08
  • Java JDK 二分法 分析demo(推薦)

    Java JDK 二分法 分析demo(推薦)

    下面小編就為大家?guī)硪黄狫ava JDK 二分法 分析demo(推薦)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-07-07
  • java對象轉(zhuǎn)型實例分析

    java對象轉(zhuǎn)型實例分析

    這篇文章主要介紹了java對象轉(zhuǎn)型的概念及用法,并以實例形式進(jìn)行了較為詳細(xì)的介紹,需要的朋友可以參考下
    2014-10-10
  • 深入探究Spring IOC和DI的區(qū)別

    深入探究Spring IOC和DI的區(qū)別

    很多人都會把ioc和di說成同一個東西,其實IOC和DI雖然在概念上可以籠統(tǒng)地視為同一事物,但其本質(zhì)上存在區(qū)別,因此,我們希望能夠更加嚴(yán)謹(jǐn)?shù)貐^(qū)分這兩個概念,以更好地理解和應(yīng)用它們,需要的朋友可以參考閱讀本文
    2023-10-10
  • SpringMVC如何域?qū)ο蠊蚕頂?shù)據(jù)

    SpringMVC如何域?qū)ο蠊蚕頂?shù)據(jù)

    在Spring MVC中,可以使用域?qū)ο髞砉蚕頂?shù)據(jù),域?qū)ο笫且粋€Map類型的對象,可以在請求處理方法之間共享數(shù)據(jù),本文給大家介紹SpringMVC 域?qū)ο蠊蚕頂?shù)據(jù)的示例代碼,一起看看吧
    2023-09-09
  • Java 給PDF簽名時添加可信時間戳的方法

    Java 給PDF簽名時添加可信時間戳的方法

    這篇文章主要介紹了Java 給PDF簽名時添加可信時間戳,關(guān)于jar導(dǎo)入的問題,本文給大家?guī)韮煞N方法,一種是手動導(dǎo)入另一種是maven配置導(dǎo)入,需要的朋友可以參考下
    2021-07-07
  • 解決websocket 報 Could not decode a text frame as UTF-8錯誤

    解決websocket 報 Could not decode a text frame as UTF-8錯誤

    這篇文章主要介紹了解決websocket 報 Could not decode a text frame as UTF-8錯誤,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-10-10
  • IDEA 控制臺中文亂碼4種解決方案

    IDEA 控制臺中文亂碼4種解決方案

    IntelliJ IDEA 如果不進(jìn)行相關(guān)設(shè)置,可能會導(dǎo)致控制臺中文亂碼、配置文件中文亂碼等問題,本文主要介紹了IDEA控制臺中文亂碼4種解決方案,具有一定的參考價值,感興趣的可以了解一下
    2024-07-07
  • Spring項目讀取配置文件中文亂碼的解決

    Spring項目讀取配置文件中文亂碼的解決

    這篇文章主要介紹了Spring項目讀取配置文件中文亂碼的解決方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-01-01

最新評論