Java之深浅拷贝

Java之深浅拷贝

基本类型:拷贝数据的值
其他类型:拷贝地址的引用,拷贝出的新对象与原对象共享实例变量,不受访问权限影响
String类型:拷贝的也是一个引用,但其内容存储在字符串池中,修改时会重新生成新的字符串,原有字符串内容是保持不变的(String是不可变的),等同于基本类型。

浅拷贝

浅拷贝就是直接使用"="给基本类型赋值(即使拷贝引用)

List<String> srcList = new ArrayList<>();
srcList.add("android");
List<String> cpyList = srcList;
srcList.add("ios");

输出这两个list,srcList和cpyList都有android,ios两项。

深拷贝

深拷贝就是将所有属性均拷贝到另外一个对象中,将其中任何一个对象的任何一个属性发生变化时,另外一个对象不受任何影响。

List<String> srcList = new ArrayList<>();
srcList.add("android");
List<String> cpyList = new ArrayList<>(srcList);
srcList.add("ios");

输出这两个list,srcList有android,ios两项,而cpyList只有android一项。

clone方法实现拷贝

每个对象都是继承于Object类的,Object类提供了clone方法,通过实现Cloneable接口可以使该具有拷贝的能力,拷贝的操作是在内存中进行中,所以性能会比直接构造对象要快速很多。

public class Student implements Cloneable {
    private String name;
    private Teacher teacher;

    public String getName() {
        return name;
    }

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

    public Teacher getTeacher() {
        return teacher;
    }

    public void setTeacher(Teacher teacher) {
        this.teacher = teacher;
    }

    @Override
    protected Student clone() {
        Student s = null;
        try {
            s = (Student) super.clone();
            s.setTeacher(s.getTeacher());
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return s;
    }

    public static void main(String[] args) {
        Student s1 = new Student();
        s1.setName("Yao");
        s1.setTeacher(new Teacher("Kobe"));
        Student s2 = s1.clone();
        s2.setName("Yi");
        s2.getTeacher().setName("Jordan");
        System.out.println(s1.getName() + "'s teacher is "
                + s1.getTeacher().getName());
        System.out.println(s2.getName() + "'s teacher is "
                + s2.getTeacher().getName());

    }
}

class Teacher {
    private String name;

    public String getName() {
        return name;
    }

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

    public Teacher(String name) {
        super();
        this.name = name;
    }
}

输出:
Yao's teacher is Jordan
Yi's teacher is Jordan
这里关于teacher的拷贝是通过"="号来操作的,仅仅实现了浅拷贝。

@Override
protected Student clone() {
    Student s = null;
    try {
        s = (Student) super.clone();
        s.setTeacher(new Teacher(s.getTeacher().getName()));
    } catch (CloneNotSupportedException e) {
        e.printStackTrace();
    }
    return s;
}

将Student类的clone方法改成这样即可实现深拷贝。
输出:
Yao's teacher is Kobe
Yi's teacher is Jordan

通过对象序列化,使用writeObject和readObject也可以实现对象的深拷贝。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,764评论 18 399
  • 对象的创建与销毁 Item 1: 使用static工厂方法,而不是构造函数创建对象:仅仅是创建对象的方法,并非Fa...
    孙小磊阅读 2,021评论 0 3
  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 11,141评论 6 13
  • 一. Java基础部分.................................................
    wy_sure阅读 3,835评论 0 11
  • 这一年,我愿意接受很多东西… 比如说: 那久违的奋斗路上彼此珍视的伙伴慢慢回来。这在以前我铁定不...
    天思静默阅读 315评论 2 11