说说深拷贝和浅拷贝

作者: 某昆 | 来源:发表于2018-08-11 15:01 被阅读6次

    本文主要内容

    • 什么是clone
    • 浅拷贝
    • 深拷贝

    1、什么是clone

    在实际编程过程中,我们常常遇到这种情况:有一个对象A,在某一时刻A中已经包含了一些有效值,此时可能 会需要一个和A完全相同新对象B,并且此后对B任何改动都不会影响到A中的值,也就是说,A与B是两个独立的对象,但B的初始值是由A对象确定的。在 Java语言中,用简单的赋值语句是不能满足这种需求的。要满足这种需求虽然有很多途径,但实现clone()方法是其中最简单,也是最高效的手段。

    Java的所有类都默认继承java.lang.Object类,在java.lang.Object类中有一个方法clone()。方法将返回一个新对象,并且新对象中将包含原来对象的信息,而不是对象的默认的初始信息。

    还有一点值得注意:

     /* @return     a clone of this instance.
     * @exception  CloneNotSupportedException  if the object's class does not
     *               support the {@code Cloneable} interface. Subclasses
     *               that override the {@code clone} method can also
     *               throw this exception to indicate that an instance cannot
     *               be cloned.
     * @see java.lang.Cloneable
     */
    protected native Object clone() throws CloneNotSupportedException;
    

    源码中关于抛出异常的说明,如果不实现 Cloneable 接口,调用 clone 方法将抛出异常。

    2、浅拷贝

    先来看一个示例,CopyUser类中包含3个成员变量,有String,int,还有一个Object对象。

    public class CopyUser implements Cloneable, Serializable{
    
    public String name;
    public int age;
    public CopyJob job;
    }
    

    重写它的clone方法:

    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
        //return deepClone1();
        //return deepClone2();
    }
    

    试试看clone方法的效果:

    public static void main(String[] args) {
        try {
            CopyJob job1 = new CopyJob("student", 0);
            CopyUser user1 = new CopyUser("jim", 15, job1);
            System.out.println(user1);
            CopyUser user2 = (CopyUser) user1.clone();
            user2.name = "tom";
            user2.age = 21;
            user2.job.jobName = "programmer";
            user2.job.salary = 1000;
            System.out.println(user2);
            System.out.println(user1);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    

    查看结果如何:

    可以看到,user2的name及aga均改变了,但user1和user2两个对象的 job 成员均变化了,它们指向了同一个 job 对象。这与我们设想的克隆不太一样。

    在 clone 方法中,对象的基础类型成员变量均会被真正克隆,在内存中生成另外一个副本,并且值和原对象的值一样,但非基础类型的成员变量不会被真正克隆,只是被赋值,指向原对象的成员变量,这就是浅拷贝 。

    要想实现深拷贝,需要我们改进下 clone 方法。

    3、深拷贝

    实现深拷贝有两个方法,第1个方法就是将非基础类型的成员变量也手动克隆一次:

    private Object deepClone1(){
        try {
            CopyJob job = (CopyJob) this.job.clone();
            CopyUser user = (CopyUser) super.clone();
            user.job = job;
            return user;
        } catch (Exception e) {
        }
        return null;
    }
    

    这种可以实现,另外还有一种方法可以实现,利用流读写Object。

    private Object deepClone2(){
        try {
            ByteArrayOutputStream bo = new ByteArrayOutputStream();
            ObjectOutputStream oo = new ObjectOutputStream(bo);
            oo.writeObject(this);
            
            ByteArrayInputStream bi = new ByteArrayInputStream(bo.toByteArray());
            ObjectInputStream oi = new ObjectInputStream(bi);
            return (oi.readObject());
        } catch (Exception e) {
        }
        return null;
    }
    

    需要强调的是,第2种方法,所涉及的所有类均要实现序列化接口,Serializable。如果不实现此接口,读到的对象将为 null。

    随便调用上面任意一个深拷贝方法,我们再来看看代码的运行结果如何:

    这下,结果正常了,深拷贝也顺利实现。

    相关文章

      网友评论

        本文标题:说说深拷贝和浅拷贝

        本文链接:https://www.haomeiwen.com/subject/nyddbftx.html