java 通过Object的clone复制对象

简介: java 通过Object的clone复制对象


java 通过Object的clone复制对象


需求背景

对象的克隆是指创建一个新的对象,且新的对象的状态与原始对象的状态相同。当对克隆的新对象进行修改时,不会影响原始对象的状态。


常规实现

image.png

运行结果

image.png

这个时候发现如果改了p2的地址属性,p1的地址属性也改变了


原因分析

出现这种情况的原因Person p2 = p1;这里将p1赋值给p2实际是将p1的引用给p2,在堆内存中p1和p2指向的是同一个对象,怎样才能实现最初的需求呢?


需求实现

因为每个类直接或间接的父类都是Object,因此它们都含有clone()方法,clone()是object类的protected 方法,所以都不能在类外进行访问。

image.png

要想对一个对象进行复制,就需要对clone方法覆盖。

同时只有类的对象自己可以克隆自己,所以对象类必须实现Cloneable接口才可以使用obj.clone()方法,

首先,如果这个对象的类不实现接口{@code Cloneable},然后{@code CloneNotSupportedException}被抛出,典型的方式:

image.png

image.png


代码实现

image.png


运行结果

image.png

此时的结果p2更改的地址属性并没有影响到p1的地址属性,需求达到。

到此为止实现的对象clone方法属于浅复制(shallow copy),那么什么是深复制(deep copy)

深复制

 大家都知道,在java语言中,分为基本数据类型和引用数据类型,基本数据类型包括(byte、short、int、long、float、double、boolean、char),引用数据类型包括(class、interface、[  ]),浅复制和深复制的主要区别就在于是否支持引用类型的成员变量的复制。

浅复制,clone()内部类似于创建一个新的对象并把对象中相应的字段通过赋值给新的对象,而引用数据类型的内容本身并不是克隆的,因此这种复制就叫浅复制。

那么


代码重现

增加Study对象

image.png

改造Person对象

image.png


运行程序

image.png


运行结果

image.png


可以看到p2更改的基础属性地址信息不影响p1的,但是p2更改的Study对象属性却会影响p1中对应的Study属性,这就是浅复制存在的问题

问题处理

在这种情况下需要对Study对象也实现Cloneable接口,同时重写clone方法覆盖

image.png


同时Person对象中需要处理Study对象的复制

image.png

这就是深复制

再次运行copy3()查看运行结果

image.png


这里可以看到p2对study属性的更改不再影响p1中study的属性。目标达成。


测试代码

TestClone.java

package com.example.demo;
import com.example.modules.person.domain.Person;
import com.example.modules.person.domain.Study;
/**
 * @author: dongao
 * @create: 2020/6/6
 */
public class TestClone {
    //测试clone
    public static void main(String[] args) {
        //copy();
        //copy2();
        copy3();
    }
    private static void copy() {
        Person p1 = new Person();
        p1.setName("小A");
        p1.setAddr("北京");
        p1.setAge(18);
        p1.setBirthday("2020-05-20");
        p1.setCountry("中国");
        System.out.println("p1-first:"+p1.getAddr());
        Person p2 = p1;
        System.out.println("p2-first:"+p2.getAddr());
        p2.setAddr("上海");
        System.out.println("p1-second:"+p1.getAddr());
        System.out.println("p2-second:"+p2.getAddr());
    }
    private static void copy2() {
        Person p1 = new Person();
        p1.setName("小A");
        p1.setAddr("北京");
        p1.setAge(18);
        p1.setBirthday("2020-05-20");
        p1.setCountry("中国");
        System.out.println("p1-first:"+p1.getAddr());
        Person p2 = p1.clone();
        System.out.println("p2-first:"+p2.getAddr());
        p2.setAddr("上海");
        System.out.println("p1-second:"+p1.getAddr());
        System.out.println("p2-second:"+p2.getAddr());
    }
    private static void copy3() {
        Person p1 = new Person();
        p1.setName("小A");
        p1.setAddr("北京");
        p1.setAge(18);
        p1.setBirthday("2020-05-20");
        p1.setCountry("中国");
        Study s1 = new Study();
        s1.setStuName("语文");
        p1.setStudy(s1);
        System.out.println("p1-first:"+p1.getAddr()+"study:"+p1.getStudy().getStuName());
        Person p2 = p1.clone();
        System.out.println("p2-first:"+p2.getAddr()+"study:"+p2.getStudy().getStuName());
        p2.setAddr("上海");
        p2.getStudy().setStuName("数学");
        System.out.println("p1-second:"+p1.getAddr()+"study:"+p1.getStudy().getStuName());
        System.out.println("p2-second:"+p2.getAddr()+"study:"+p2.getStudy().getStuName());
    }
}


Person.java

package com.example.modules.person.domain;
public class Person implements Cloneable
{
  private String id;
  private String name;
  private Integer age;
  private String country;
  private String birthday;
  private String addr;
  private String data;
  public String getId() {
    return id;
  }
  public void setId(String id) {
    this.id = id;
  }
  public String getName() {
    return name;
  }
  public void setName(String name) {
    this.name = name;
  }
  public Integer getAge() {
    return age;
  }
  public void setAge(Integer age) {
    this.age = age;
  }
  public String getCountry() {
    return country;
  }
  public void setCountry(String country) {
    this.country = country;
  }
  public String getBirthday() {
    return birthday;
  }
  public void setBirthday(String birthday) {
    this.birthday = birthday;
  }
  public String getAddr() {
    return addr;
  }
  public void setAddr(String addr) {
    this.addr = addr;
  }
  public String getData() {
    return data;
  }
  public void setData(String data) {
    this.data = data;
  }
  private Study study;
  public Study getStudy() {
    return study;
  }
  public void setStudy(Study study) {
    this.study = study;
  }
  @Override
  public Person clone() {
    Person per = null;
    try {
      per = (Person) super.clone();
      per.study = study.clone();
    } catch (CloneNotSupportedException e) {
      e.printStackTrace();
    }
    return per;
  }
}


Study.java

package com.example.modules.person.domain;
/**
 * @author: dongao
 * @create: 2020/6/6
 */
public class Study implements Cloneable{
    private String stuName;
    public String getStuName() {
        return stuName;
    }
    public void setStuName(String stuName) {
        this.stuName = stuName;
    }
    @Override
    public Study clone(){
        Study study = null;
        try {
             study = (Study) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return study;
    }
}


流方式复制对象

到这里基本问题都解决了,但是还可能会遇到一个问题,就是当前对象中有很多引用对象,这样的话通过clone的方法处理起来会比较麻烦,这是可以采用另外一种方式,序列化的方式来实现对象的深复制。


改造Person.java

image.png


这里最好显式指定serialVersionUID的值防止反序列化的时候出问题。


image.png

  public Person pclone() {
    Person per = null;
    try {
      //对象的序列化流,作用:把对象转成字节数据的输出到文件中保存,对象的输出过程称为序列化,可实现对象的持久存储。
      ByteArrayOutputStream bout = new ByteArrayOutputStream();
      ObjectOutputStream out = new ObjectOutputStream(bout);
      out.writeObject(this);
      out.close();
      // read a clone of the object from the byte array
      ByteArrayInputStream bin = new ByteArrayInputStream(bout.toByteArray());
      ObjectInputStream in = new ObjectInputStream(bin);
      per = (Person) in.readObject();
      in.close();
    } catch (Exception e) {
      e.printStackTrace();
    }
    return per;
  }


运行效果

image.png

**注:**大家可以自行复制代码测试,欢迎指正,相互学习。


相关文章
|
1月前
|
安全 Java 编译器
Java对象一定分配在堆上吗?
本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。
Java对象一定分配在堆上吗?
|
2月前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
50 17
|
1月前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
2月前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第7天】Java零基础教学篇,手把手实践教学!
33 6
|
1月前
|
存储 缓存 NoSQL
一篇搞懂!Java对象序列化与反序列化的底层逻辑
本文介绍了Java中的序列化与反序列化,包括基本概念、应用场景、实现方式及注意事项。序列化是将对象转换为字节流,便于存储和传输;反序列化则是将字节流还原为对象。文中详细讲解了实现序列化的步骤,以及常见的反序列化失败原因和最佳实践。通过实例和代码示例,帮助读者更好地理解和应用这一重要技术。
44 0
|
1月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
117 4
|
2月前
|
Java
Java Object 类详解
在 Java 中,`Object` 类是所有类的根类,每个 Java 类都直接或间接继承自 `Object`。作为所有类的超类,`Object` 定义了若干基本方法,如 `equals`、`hashCode`、`toString` 等,这些方法在所有对象中均可使用。通过重写这些方法,可以实现基于内容的比较、生成有意义的字符串表示以及确保哈希码的一致性。此外,`Object` 还提供了 `clone`、`getClass`、`notify`、`notifyAll` 和 `wait` 等方法,支持对象克隆、反射机制及线程同步。理解和重写这些方法有助于提升 Java 代码的可读性和可维护性。
103 20
|
4月前
|
Java
【Java基础面试二十】、介绍一下Object类中的方法
这篇文章介绍了Java中Object类的常用方法,包括`getClass()`、`equals()`、`hashCode()`、`toString()`、`wait()`、`notify()`、`notifyAll()`和`clone()`,并提到了不推荐使用的`finalize()`方法。
【Java基础面试二十】、介绍一下Object类中的方法
|
3月前
|
Python
类与面向对象编程(Object-Oriented Programming, OOP)
类与面向对象编程(Object-Oriented Programming, OOP)
25 0
|
4月前
|
前端开发 Java 编译器
【前端学java】java中的Object类和前端中的Object有什么区别(9)
【8月更文挑战第10天】java中的Object类和前端中的Object有什么区别
49 0
【前端学java】java中的Object类和前端中的Object有什么区别(9)
下一篇
DataWorks