java 浅克隆和深克隆

2017-04-01  本文已影响0人  那脸憔悴

在使用克隆时,我们需要知道使用的目的:就是为了快速构造一个和已有对象相同的副本。
一、浅克隆:
要实现java.lang.Cloneable接口,并重写clone方法。

public class A implements Cloneable {
    private String name;
    private int age;
    private int[] s;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public int[] getS() {
        return s;
    }
    public void setS(int[] s) {
        this.s = s;
    }
    @Override
    protected Object clone() throws CloneNotSupportedException {
        
        return super.clone();
    }
}

测试类:

public class Test {
    public static void main(String[] args) throws CloneNotSupportedException {
        A a = new A();
        a.setName("abc");
        a.setAge(22);
        a.setS(new int[]{12});
        System.out.println("克隆前a:name="+a.getName()+" age="+a.getAge()+" s="+a.getS()[0]);
        A a2 = (A) a.clone();
        a2.setName("aaa");
        a2.setAge(10);
        int[] i = a2.getS();
        i[0] = 13;
        a2.setS(i);
        System.out.println("克隆后a:name="+a.getName()+" age="+a.getAge()+" s="+a.getS()[0]);
        System.out.println("克隆后a2:name="+a2.getName()+" age="+a2.getAge()+" s="+a2.getS()[0]);
    }
}
打印结果:
克隆前a:name=abc age=22 s=12
克隆后a:name=abc age=22 s=13
克隆后a2:name=aaa age=10 s=13

可以看出,基本类型可以使用浅克隆,而对于引用类型,由于引用的是内容相同,所以改变a2实例对象中的属性就会影响到a。所以引用类型需要使用深克隆。
二、使用序列化实现深克隆:
javabean类

import java.io.Serializable;
public class B implements Serializable {
    private static final long serialVersionUID = 1L;
    private String name;
    private int age;
    private int[] s;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public int[] getS() {
        return s;
    }
    public void setS(int[] s) {
        this.s = s;
    }
}

克隆类

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
public class DeepClone {
    public Object deepClone(Object src) {
        Object o = null;
        try {
            if (src != null) {
                ByteArrayOutputStream baos = new ByteArrayOutputStream();
                ObjectOutputStream oos = new ObjectOutputStream(baos);
                oos.writeObject(src);
                ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());
                ObjectInputStream ois = new ObjectInputStream(bais);
                o = ois.readObject();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
        return o;
    }
}

测试类

public class Test {
    public static void main(String[] args) throws CloneNotSupportedException {
        B b = new B();
        b.setName("abc");
        b.setAge(22);
        b.setS(new int[]{12});
        System.out.println("克隆前a:name="+b.getName()+" age="+b.getAge()+" s="+b.getS()[0]);
        DeepClone dc = new DeepClone();
        B b2 = (B) dc.deepClone(b);
        b2.setName("aaa");
        b2.setAge(20);
        int[] i2 = b2.getS();
        i2[0] = 10;
        b2.setS(i2);
        System.out.println("克隆后a:name="+b.getName()+" age="+b.getAge()+" s="+b.getS()[0]);
        System.out.println("克隆后a2:name="+b2.getName()+" age="+b2.getAge()+" s="+b2.getS()[0]);
    }
}
打印结果:
克隆前a:name=abc age=22 s=12
克隆后a:name=abc age=22 s=12
克隆后a2:name=aaa age=20 s=10

可以看到,两个引用所指向的对象在堆中相互独立,互不干扰,这样就实现了深度克隆。

上一篇下一篇

猜你喜欢

热点阅读