详解java中的深拷贝和浅拷贝(clone()方法的重写、使用序列化实现真正的深拷贝)

1.序列化实现

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("--------使用序列化实现对象的拷贝--------");
    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());
  }

  /*使用默认的clone方法,需要Person实现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+"】");
    //改一个就会触动全部!! 这就是使用默认的clone方法的弊端
    /*该clone()方法是使用Object类的clone()方法,但是该方法存在一个缺陷,它并不会将对象的所有属性全部拷贝过来,而是有选择性的拷贝,基本规则如下:
       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+"】");
  }

  /*使用序列化实现对象的拷贝,需要对象以及对象中的其他对象都要实现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 : 我】

--------使用序列化实现对象的拷贝--------
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()方法的重写、使用序列化实现真正的深拷贝)详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

(0)

相关推荐

  • 浅谈Java序列化和hessian序列化的差异

    在远程调用中,需要把参数和返回值通过网络传输,这个使用就要用到序列化将对象转变成字节流,从一端到另一端之后再反序列化回来变成对象. 既然前面有一篇提到了hessian,这里就简单讲讲Java序列化和hessian序列化的区别. 首先,hessian序列化比Java序列化高效很多,而且生成的字节流也要短很多.但相对来说没有Java序列化可靠,而且也不如Java序列化支持的全面.而之所以会出现这样的区别,则要从它们的实现方式来看. 先说Java序列化,具体工作原理就不说了,Java序列化会把要序列化

  • java IO数据操作流、对象序列化、压缩流代码解析

    数据操作流 在io包中,提供了两个与平台无关的数据操作流: 数据输入流(DataInputStream) 数据输出流(DataOutputStream) 通常数据输出流会按一定格式将数据输出,再通过数据输入流按照一定格式将数据读入 DataOutputStream接口定义了一系列的writeXxx()的操作,可以写入各种数据类型的数据. 范例:使用数据操作流写入与读出数据 import java.io.DataOutputStream ; import java.io.File ; import

  • Java对象的XML序列化与反序列化实例解析

    上一篇文章我们介绍了java实现的各种排序算法代码示例,本文我们看看Java对象的xml序列化与反序列化的相关内容,具体如下. XML是一种标准的数据交换规范,可以方便地用于在应用之间交换各类数据.如果能在Java对象和XML文档之间建立某种映射,例如Java对象的XML序列化和反序列化,那么就可以使Java的对象方便地与其他应用进行交换. java.beans包里面有两个类XMLEncoder和Decoder,分别用于将符合JabaBeans规范的Java对象以XML方式序列化和反序列化.以下

  • 解决Spring Boot和Feign中使用Java 8时间日期API(LocalDate等)的序列化问题

    LocalDate . LocalTime . LocalDateTime 是Java 8开始提供的时间日期API,主要用来优化Java 8以前对于时间日期的处理操作.然而,我们在使用Spring Boot或使用Spring Cloud Feign的时候,往往会发现使用请求参数或返回结果中有 LocalDate . LocalTime . LocalDateTime 的时候会发生各种问题.本文我们就来说说这种情况下出现的问题,以及如何解决. 问题现象 先来看看症状.比如下面的例子: @Sprin

  • java 中序列化NotSerializableException问题解决办法

    java 中序列化NotSerializableException问题解决办法 前言: 某项目中,要将某个自定义类MMessage对象,通过ObjectOutputStream和ObjectInputStream传递,该MMessage的特征描述: 1 该类未继承Serializable接口: 2 其父类Message的父类继承了Serializable接口: 3 其父类中有一个字段类型为Java.io.ByteArrayOutputStream类型: 经测试发现,MMessage类序列化过程中

  • Java对象序列化操作详解

    本文实例讲述了Java对象序列化操作.分享给大家供大家参考,具体如下: 当两个进程在进行远程通信时,彼此可以发送各种类型的数据.无论是何种类型的数据,都会以二进制序列的形式在网络上传送.发送方需要把这个Java对象转换为字节序列,才能在网络上传送:接收方则需要把字节序列再恢复为Java对象. 只能将支持 java.io.Serializable 接口的对象写入流中.每个 serializable 对象的类都被编码,编码内容包括类名和类签名.对象的字段值和数组值,以及从初始对象中引用的其他所有对象

  • java原生序列化和Kryo序列化性能实例对比分析

    简介 最近几年,各种新的高效序列化方式层出不穷,不断刷新序列化性能的上限,最典型的包括: 专门针对Java语言的:Kryo,FST等等 跨语言的:Protostuff,ProtoBuf,Thrift,Avro,MsgPack等等 这些序列化方式的性能多数都显著优于hessian2(甚至包括尚未成熟的dubbo序列化).有鉴于此,我们为dubbo引入Kryo和FST这 两种高效Java序列化实现,来逐步取代hessian2.其中,Kryo是一种非常成熟的序列化实现,已经在Twitter.Group

  • 通过实例了解java序列化机制

    这篇文章主要介绍了通过实例了解java序列化机制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 序列化是指对象通过写出描述自己状态的数值来记录自己的过程,即将对象表示成一系列有序字节,Java提供了将对象写入流和从流中恢复对象的方法.对象能包含其它的对象,而其它的对象又可以包含另外的对象.Java序列化能够自动的处理嵌套的对象.对于一个对象的简单域,writeObject()直接将其值写入流中. 当遇到一个对象域时,writeObject()被

  • 详解Java中的println输入和toString方法的重写问题

    目录 println()直接打印 toString的使用 其他的一些情况 println()直接打印 我们都知道println()如果打印的是基本数据类型的话直接打印出来的就是值,你如果是引用数据类型呢? 除掉这四类八种基本类型,其它的都是对象,也就是引用类型,包括数组. 让我们来测试一下 public class test1 { public static void main(String[] args) { int[] ret = {1, 2, 3, 4}; System.out.print

  • 详解Java中List的正确的删除方法

    目录 简介 实例 正确方法 法1:for的下标倒序遍历 法2: list.stream().filter().collect() 法3: iterator迭代器 错误方法 法1:for(xxx : yyy)遍历 法2:for的下标正序遍历 原因分析 简介 本文介绍Java的List的正确的删除方法. 实例 需求:有如下初始数据,将list中的所有数据为"b"的元素删除掉.即:填充removeB()方法 package com.example.a; import java.util.Ar

  • 详解Java中异步转同步的六种方法

    目录 一.问题 应用场景 二.分析 三.实现方法 1.轮询与休眠重试机制 2.wait/notify 3.Lock Condition 4.CountDownLatch 5.CyclicBarrier 6.LockSupport 一.问题 应用场景 应用中通过框架发送异步命令时,不能立刻返回命令的执行结果,而是异步返回命令的执行结果. 那么,问题来了,针对应用中这种异步调用,能不能像同步调用一样立刻获取到命令的执行结果,如何实现异步转同步? 二.分析 首先,解释下同步和异步 同步,就是发出一个调

  • 详解Java中@Override的作用

    详解Java中@Override的作用 @Override是伪代码,表示重写(当然不写也可以),不过写上有如下好处: 1.可以当注释用,方便阅读: 2.编译器可以给你验证@Override下面的方法名是否是你父类中所有的,如果没有则报错.例如,你如果没写@Override,而你下面的方法名又写错了,这时你的编译器是可以编译通过的,因为编译器以为这个方法是你的子类中自己增加的方法. 举例:在重写父类的onCreate时,在方法前面加上@Override 系统可以帮你检查方法的正确性. @Overr

  • 详解Java中AbstractMap抽象类

    jdk1.8.0_144 下载地址:http://www.jb51.net/softs/551512.html AbstractMap抽象类实现了一些简单且通用的方法,本身并不难.但在这个抽象类中有两个方法非常值得关注,keySet和values方法源码的实现可以说是教科书式的典范. 抽象类通常作为一种骨架实现,为各自子类实现公共的方法.上一篇我们讲解了Map接口,此篇对AbstractMap抽象类进行剖析研究. Java中Map类型的数据结构有相当多,AbstractMap作为它们的骨架实现实

  • 详解Java中的不可变对象

    不可变对象想必大部分朋友都不陌生,大家在平时写代码的过程中100%会使用到不可变对象,比如最常见的String对象.包装器对象等,那么到底为何Java语言要这么设计,真正意图和考虑点是什么?可能一些朋友没有细想过这些问题,今天我们就来聊聊跟不可变对象有关的话题. 一.什么是不可变对象 下面是<Effective Java>这本书对于不可变对象的定义: 不可变对象(Immutable Object):对象一旦被创建后,对象所有的状态及属性在其生命周期内不会发生任何变化. 从不可变对象的定义来看,

  • 图文详解Java中的序列化机制

    目录 概述 对象序列化和反序列化机制 修改默认的序列化机制 使用transient关键字 自定义readObject.writeObject方法 实现Externalizable接口 serialVersionUID的作用 使用序列化clone 概述 java中的序列化可能大家像我一样都停留在实现Serializable接口上,对于它里面的一些核心机制没有深入了解过.直到最近在项目中踩了一个坑,就是序列化对象添加一个字段以后,使用方系统报了反序列化失败,原因是我们双方的序列化对象没有加上seri

  • 详解Java中多线程异常捕获Runnable的实现

    详解Java中多线程异常捕获Runnable的实现 1.背景: Java 多线程异常不向主线程抛,自己处理,外部捕获不了异常.所以要实现主线程对子线程异常的捕获. 2.工具: 实现Runnable接口的LayerInitTask类,ThreadException类,线程安全的Vector 3.思路: 向LayerInitTask中传入Vector,记录异常情况,外部遍历,判断,抛出异常. 4.代码: package step5.exception; import java.util.Vector

  • 详解java 中Spring jsonp 跨域请求的实例

    详解java 中Spring jsonp 跨域请求的实例 jsonp介绍 JSONP(JSON with Padding)是JSON的一种"使用模式",可用于解决主流浏览器的跨域数据访问的问题.由于同源策略,一般来说位于 server1.example.com 的网页无法与不是 server1.example.com的服务器沟通,而 HTML 的<script> 元素是一个例外.利用 <script> 元素的这个开放策略,网页可以得到从其他来源动态产生的 JSO

  • 详解Java 中的嵌套类与内部类

    详解Java 中的嵌套类与内部类 在Java中,可以在一个类内部定义另一个类,这种类称为嵌套类(nested class).嵌套类有两种类型:静态嵌套类和非静态嵌套类.静态嵌套类较少使用,非静态嵌套类使用较多,也就是常说的内部类.其中内部类又分为三种类型: 1.在外部类中直接定义的内部类. 2.在函数中定义的内部类. 3.匿名内部类. 对于这几种类型的访问规则, 示例程序如下: package lxg; //定义外部类 public class OuterClass { //外部类静态成员变量

随机推荐