浅拷贝小结:
Object对象有个clone()方法,实现了对象中各个属性的复制,但它的可见范围是protected的,所以实体类使用克隆的前提是:
① 实现Cloneable接口,这是一个标记接口,自身没有方法。
② 覆盖clone()方法,可见性提升为public。
如果一个被复制的属性都是基本类型,那么只需要实现当前类的cloneable机制就可以了,此为浅拷贝。
如果被复制对象的属性包含其他实体类对象引用,那么这些实体类对象都需要实现cloneable接口并覆盖clone()方法。

深拷贝小结
① 如果有一个非原生成员,如自定义对象的成员,那么就需要:该成员实现Cloneable接口并覆盖clone()方法,不要忘记提升为public可见。同时,修改被复制类的clone()方法,增加成员的克隆逻辑。
② 如果被复制对象不是直接继承Object,中间还有其它继承层次,每一层super类都需要实现Cloneable接口并覆盖clone()方法。
与对象成员不同,继承关系中的clone不需要被复制类的clone()做多余的工作。
一句话来说,如果实现完整的深拷贝,需要被复制对象的继承链、引用链上的每一个对象都实现克隆机制。

浅拷贝: 被复制对象的所有值属性都含有与原来对象的相同,而所有的对象引用属性仍然指向原来的对象。

package com.basis.entity;
 
public class Person implements Cloneable {
 
    private String adress;
    private Info info;
 
    public String getAdress() {
        return adress;
    }
 
    public void setAdress(String adress) {
        this.adress = adress;
    }
 
    public Info getInfo() {
        return info;
    }
 
    public void setInfo(Info info) {
        this.info = info;
    }
 
    @Override
    public String toString() {
        return "Person{" +
                "adress='" + adress + '\'' +
                ", info=" + info +
                '}';
    }
 
    public Object clone() throws CloneNotSupportedException{
        return super.clone();
    }
}
package com.basis.entity;
 
public class Info {
 
    private String IDCard;
    private String name;
 
    public String getIDCard() {
        return IDCard;
    }
 
    public void setIDCard(String IDCard) {
        this.IDCard = IDCard;
    }
 
    public String getName() {
        return name;
    }
 
    public void setName(String name) {
        this.name = name;
    }
 
    @Override
    public String toString() {
        return "Info{" +
                "IDCard='" + IDCard + '\'' +
                ", name='" + name + '\'' +
                '}';
    }
}
package com.company;
 
import com.basis.entity.Info;
import com.basis.entity.Person;
 
public class Main {
 
    public static void main(String[] args) {
 
        Info info = new Info();
        info.setName("张三");
        info.setIDCard("111111");
 
        Person person1 = new Person();
        person1.setAdress("北京");
        person1.setInfo(info);
 
        System.out.println("clone()前person1:"+person1.toString());
 
        Person person2 = null;
        try {
            person2 = (Person)person1.clone();
            person2.setAdress("河南");
            Info info1 = person2.getInfo();
            info1.setName("李四");
            info1.setIDCard("222222");
            person2.setInfo(info1);
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        System.out.println("clone()后person1:"+person1.toString());
        System.out.println("clone()后person2:"+person2.toString());
 
    }
}

结果

clone()前person1:Person{adress='北京', info=Info{IDCard='111111', name='张三'}}
clone()后person1:Person{adress='北京', info=Info{IDCard='222222', name='李四'}}
clone()后person2:Person{adress='河南', info=Info{IDCard='222222', name='李四'}}

现象:
修改clone()后的那个对象person2的adress变量时,person1的adress变量没有发生修改。
但是修改clone()后的那个对象person2的Info变量时,而person1的Info变量也相应修改。

说明 :
当修改clone()后的对象时,修改的字段为引用类型,而不是基础数据类型时,会对clone()前的对象有所影响,
而这种实现的方式为浅拷贝(shadow copy)。

为了解决修改clone()得到的新对象不会影响被复制对象的字段的问题,我们需要实现深拷贝(deep copy)。

**深拷贝:**在浅拷贝的基础上,所有引用其他对象的变量也进行了clone,并指向被复制过的新对象

package com.basis.entity;
 
public class Person implements Cloneable {
 
    private String adress;
    private Info info;
 
    public String getAdress() {
        return adress;
    }
 
    public void setAdress(String adress) {
        this.adress = adress;
    }
 
    public Info getInfo() {
        return info;
    }
 
    public void setInfo(Info info) {
        this.info = info;
    }
 
    @Override
    public String toString() {
        return "Person{" +
                "adress='" + adress + '\'' +
                ", info=" + info +
                '}';
    }
 
    public Object clone() throws CloneNotSupportedException{
        Person person = (Person) super.clone();
        person.info = (Info) info.clone();
        return person;
    }
}
package com.basis.entity;
 
public class Info implements Cloneable {
 
    private String IDCard;
    private String name;
 
    public String getIDCard() {
        return IDCard;
    }
 
    public void setIDCard(String IDCard) {
        this.IDCard = IDCard;
    }
 
    public String getName() {
        return name;
    }
 
    public void setName(String name) {
        this.name = name;
    }
 
    @Override
    public String toString() {
        return "Info{" +
                "IDCard='" + IDCard + '\'' +
                ", name='" + name + '\'' +
                '}';
    }
 
    public Object clone() throws CloneNotSupportedException{
        return super.clone();
    }
}
package com.company;
 
import com.basis.entity.Info;
import com.basis.entity.Person;
 
public class Main {
 
    public static void main(String[] args) {
 
        Info info = new Info();
        info.setName("张三");
        info.setIDCard("111111");
 
        Person person1 = new Person();
        person1.setAdress("北京");
        person1.setInfo(info);
 
        System.out.println("clone()前person1:"+person1.toString());
 
        Person person2 = null;
        try {
            person2 = (Person)person1.clone();
            person2.setAdress("河南");
            Info info1 = person2.getInfo();
            info1.setName("李四");
            info1.setIDCard("222222");
            person2.setInfo(info1);
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        System.out.println("clone()后person1:"+person1.toString());
        System.out.println("clone()后person2:"+person2.toString());
 
    }
}

结果

clone()前person1:Person{adress='北京', info=Info{IDCard='111111', name='张三'}}
clone()后person1:Person{adress='北京', info=Info{IDCard='111111', name='张三'}}
clone()后person2:Person{adress='河南', info=Info{IDCard='222222', name='李四'}}