深复制和浅复制

⑴浅复制(浅克隆)
被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引
用仍然指向原来的对象。换言之,浅复制仅仅复制所考虑的对象,而不复制它所
引用的对象。

⑵深复制(深克隆)
被复制对象的所有变量都含有与原来的对象相同的值,除去那些引用其他对象的
变量。那些引用其他对象的变量将指向被复制过的新对象,而不再是原有的那些
被引用的对象。换言之,深复制把要复制的对象所引用的对象都复制了一遍。

2.Java的clone()方法
⑴clone方法将对象复制了一份并返回给调用者。一般而言,clone()方法满足:
①对任何的对象x,都有x.clone() !=x//克隆对象与原对象不是同一个对象
②对任何的对象x,都有x.clone().getClass()= =x.getClass()//克隆对象与原对象的类型一样
③如果对象x的equals()方法定义恰当,那么x.clone().equals(x)应该成立。

⑵Java中对象的克隆
①为了获取对象的一份拷贝,我们可以利用Object类的clone()方法。
②在派生类中覆盖基类的clone()方法,并声明为public。
③在派生类的clone()方法中,调用super.clone()。
④在派生类中实现Cloneable接口。

-----------------------------------------------------------

浅拷贝:只复制一个对象,对象内部存在的指向其他对象数组或者引用则不复制
深拷贝:对象,对象内部的引用均复制
为了更好的理解它们的区别我们假设有一个对象A,它包含有2对象对象A1和对象A2

clone

对象A进行浅拷贝后,得到对象B但是对象A1和A2并没有被拷贝

cloneA

对象A进行深拷贝,得到对象B的同时A1和A2连同它们的引用也被拷贝
deepClone
在理解了深拷贝和浅拷贝后,我们来看看Java的深拷贝和浅拷贝实现。
java.lang.Object的clone()方法默认是返回一个浅拷贝对象。因此如果要用clone()方法实现一个深拷贝,我们必须对每个对象的clone()方法进行特别实现。当对象层次复杂的时候,这样做不但困难而且浪费时间和容易出现错误,特别有时候你不但需要深拷贝同时你也对这个对象进行浅拷贝的时候,你会发现写这个clone()方法真不是一个好的解决方案。

除了clone()方法,还可以使用序列化来实现拷贝操作。把要拷贝的对象输出成
byte array,然后再利用ObjectInputStream转换出新的对象


public class Test implements Serializable{

    private String name = "hello";

    protected Object clone(Object obj){
        Object object = null;
        try {
            ByteArrayOutputStream out = new ByteArrayOutputStream();
            ObjectOutputStream outputStream = new ObjectOutputStream(out);
            outputStream.writeObject(obj);
            outputStream.flush();
            outputStream.close();

            ByteArrayInputStream in = new ByteArrayInputStream(out.toByteArray());
            ObjectInputStream inputStream = new ObjectInputStream(in);

            object = inputStream.readObject();
        } catch (IOException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
        return object;
    }

    public static void main(String args[]){
        Test test = new Test();
        Test one = (Test)test.clone(test);
        one.name = "world";
        System.out.print(test.name + " " + one.name);
    }
}

你可能感兴趣的:(复制)