Java 克隆,浅拷贝,深拷贝

克隆

当使用“=”赋值引用类型的时候,其实是直接赋值该对象的地址,本质还是一个对象。

而克隆则是完全创造一个新的对象出来,有自己的新地址,只是初始化的数据相同。

克隆属于浅拷贝。

Person p1 = new Person("Tom", 11);
Person p2 = p1;
Person p3 = (Person) p1.clone();
System.out.println(p1);
System.out.println(p2);
System.out.println(p3);

输出信息:

Person@74a14482
Person@74a14482
Person@1540e19d

浅拷贝

对于对象或数组类型,将对象A赋值给对象B,然后改变对象B的属性,对象A也会随之变化。

对象A对象B指向同一块内存,浅拷贝就是拷贝内存地址。

需要实现Cloneable接口,重写clone()方法。

class Person implements Cloneable {
    public String name;
    public int age;
    public Address addr;

    public Person(String name, int age, Address addr) {
        this.name = name;
        this.age = age;
        this.addr = addr;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", addr=" + addr +
                '}';
    }

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

    public static class Address {
        public String address;

        public Address(String address) {
            this.address = address;
        }

        @Override
        public String toString() {
            return "Address{" +
                    "address='" + address + '\'' +
                    '}';
        }
    }
}
Person p1 = new Person("Tom", 11, new Person.Address("beijing"));
Person p2 = (Person) p1.clone();
p2.addr.address = "shanghai";

System.out.println(p1);
System.out.println(p2);

输出信息:

Person{name='Tom', age=11, addr=Address{address='shanghai'}}
Person{name='Tom', age=11, addr=Address{address='shanghai'}}

深拷贝

深拷贝是指对象B开辟新的内存,将对象A的各个属性都复制到新内存中,

对象B的属性发生变化时,对象A的属性不会受影响。

深拷贝相比于浅拷贝速度较慢并且花销较大。

比较常用的方案有两种:

  1. 继续使用clone()方法对其内部的引用类型,再进行一次clone()
  2. 序列化这个对象,再反序列回来,重新获取这个新对象。
使用Cloneable接口
class Person implements Cloneable {
    public String name;
    public int age;
    public Address addr;

    public Person(String name, int age, Address addr) {
        this.name = name;
        this.age = age;
        this.addr = addr;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", addr=" + addr +
                '}';
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Person newPerson = (Person) super.clone();
        newPerson.addr = (Address) addr.clone();
        return newPerson;
    }

    public static class Address implements Cloneable {
        public String address;

        public Address(String address) {
            this.address = address;
        }

        @Override
        public String toString() {
            return "Address{" +
                    "address='" + address + '\'' +
                    '}';
        }

        @Override
        protected Object clone() throws CloneNotSupportedException {
            return super.clone();
        }
    }
}
Person p1 = new Person("Tom", 11, new Person.Address("beijing"));
Person p2 = (Person) p1.clone();
p2.addr.address = "shanghai";

System.out.println(p1);
System.out.println(p2);

输出信息:

Person{name='Tom', age=11, addr=Address{address='beijing'}}
Person{name='Tom', age=11, addr=Address{address='shanghai'}}

其实和浅复制是一样的道理,在深复制时将Address也进行了一次深复制。

使用反序列化
class Person implements Serializable {
    public String name;
    public int age;
    public Address addr;

    public Person(String name, int age, Address addr) {
        this.name = name;
        this.age = age;
        this.addr = addr;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", addr=" + addr +
                '}';
    }

    public static class Address implements Serializable {
        public String address;

        public Address(String address) {
            this.address = address;
        }

        @Override
        public String toString() {
            return "Address{" +
                    "address='" + address + '\'' +
                    '}';
        }
    }
}
class CloneUtils {

    public static <T extends Serializable> T deepClone(T obj) {
        T cloneObj = null;
        ByteArrayOutputStream baos = null;
        ObjectOutputStream oos = null;
        ByteArrayInputStream bais = null;
        ObjectInputStream ois = null;
        try {
            baos = new ByteArrayOutputStream();
            oos = new ObjectOutputStream(baos);
            oos.writeObject(obj);

            bais = new ByteArrayInputStream(baos.toByteArray());
            ois = new ObjectInputStream(bais);
            cloneObj = (T) ois.readObject();
        } catch (IOException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } finally {
            if (oos != null) {
                try {
                    oos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (bais != null) {
                try {
                    bais.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (ois != null) {
                try {
                    ois.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (baos != null) {
                try {
                    baos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
        return cloneObj;
    }
}
Person p1 = new Person("Tom", 11, new Person.Address("beijing"));
Person p2 = CloneUtils.deepClone(p1);
p2.name = "Jim";
p2.addr.address = "shanghai";

System.out.println(p1);
System.out.println(p2);

输出信息:

Person{name='Tom', age=11, addr=Address{address='beijing'}}
Person{name='Jim', age=11, addr=Address{address='shanghai'}}
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值