Bootstrap

Clone使用

一)克隆接口的说明:

一个对象要想实现克隆,实现克隆接口:clone方法是默认是Object的方法

1)这个接口是一个标记性的接口(空接口),内部都没有方法和属性,实现这个接口表示这个对象都可以进行克隆,调用Object对象的Object.clone()方法,如果没有实现Cloneable的类对象调用clone()就会抛出CloneNotSupportedException异常

2)想要一个类具备拷贝实例的功能,除了要实现Cloneable接口,还必须要重写Object类的clone()方法

3)我们可以看到Object类的clone()方法是被protected修饰的,我们需要在重写的clone()方法通过super关键字调用Object的clone()方法,

4)Cloneable接口发挥的是标记功能,自定义类型需要用户自己去标记哪些类是可以被clone的,这个标记就是去实现Cloneable接口,实现该接口的类就表示该类创建的对象可以被克隆

5)想要克隆自定义类型,就要实现克隆接口public interface Cloneable(){};空接口也叫作标记接口

二)浅拷贝:

浅拷贝:在拷贝一个对象的时候,只是对对象的基本数据类型的成员变量进行拷贝,但是对引用类型的成员变量只进行引用的传递,针对引用数据类型来说并没有创建一个新的对象,当对引用类型的修改会影响到要拷贝的对象,简而言之,浅拷贝只是复制所拷贝的地址,而不复制他的所引用的对象


@Data
class Student implements Cloneable{
    public String username;
    public String password;
    public Student(String username, String password) {
        this.username = username;
        this.password = password;
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Money money=(Money)super.clone();
    }
}
public class HelloWorld{
    public static void main(String[] args) throws CloneNotSupportedException {
        Student student1=new Student("李佳伟","12503487");
        Student student2= (Student) student1.clone();
        System.out.println(student1==student2);
//false,两个引用分别指向的对象的地址值不相同
    }
}

false
Student{username='李佳伟', password='12503487'}
Student{username='李佳伟', password='12503487'}
 

上面画的图就是克隆出来了一个新的对象和新的引用,但是由于这个对象的组成全部是一个基本数据类型,所以只对对象中的基本数据类型复制了一份

class Money{
    public double m=100.5;
        }
class Student implements Cloneable
{
    String name;
//实现这个接口要重写克隆方法
Money money=new Money();
    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

public class Tomact {
    public static void main(String[] args) throws CloneNotSupportedException {
        Student student1=new Student();
        Student student2= (Student) student1.clone();
        System.out.println(student1.money.m);
        System.out.println(student2.money.m);
        student2.money.m=199;
        System.out.println(student1.money.m);
        System.out.println(student2.money.m);
//都变成了199
    }
}

1)此时单纯的修改student1的username此时是不会修改student2的username的

2) 我们观察上面的代码,我们将Money类的实例引用作为了Student类成员变量的一个字段,此时克隆操作将这个student1成员变量的Money引用地址拷贝到student2对象里面,这里面的拷贝就是浅拷贝了,只是将student1的对象的money的引用进行拷贝了一份,此时student1和student2对象中的两个相同的money引用指向的是同一个对象,因为只是拷贝了一份地址

3)由此看出,只是拷贝了money对象的引用的地址,但是这两个引用仍然指向的是同一个对象,此时原来的对象和新被克隆出来的对象的成员变量money指向地址的都是 0x100

三)深拷贝以及实现: 

class Money implements Cloneable{
    public double m=100.5;
    @Override
    protected Object clone() throws CloneNotSupportedException {
        return  super.clone();
    }
}
class Student implements Cloneable
{
    String name;
    //实现这个接口要重写克隆方法
    Money money=new Money();
    @Override
    protected Object clone() throws CloneNotSupportedException {

        //return super.clone();正常情况下,Person的克隆是在这里面执行
        //Student s=(Student) super.clone();
        //return s;
        //1克隆Person
        Student s= (Student)super.clone();
        //2克隆当前的Money对象
        s.money=(Money)this.money.clone();
        return s;
        // 由于下面要进行Money的克隆
    }
}

public class Tomact {
    public static void main(String[] args) throws CloneNotSupportedException {
        Student student1=new Student();
        Student student2= (Student) student1.clone();
        System.out.println(student1.money.m);
        System.out.println(student2.money.m);
        student2.money.m=199;
        System.out.println(student1.money.m);
        System.out.println(student2.money.m);
        //这里面只是克隆了一个person,我们要想实现深拷贝
        //就要把Student对象中的money这个对象也克隆一份
    }
}

Java中的clone方法是一个浅拷贝,引用类型仍然在传递引用,可以通过继续调用clone方法,对该对象的引用类型变量在实现一次clone的方法来实现深拷贝

1)让我们的两个类都实现cloneable接口

2)在我们的Student类里面不光克隆Student对象里面的属性,还进行克隆Student类中的引用类型所指向的对象也进行克隆一份

class Money implements Cloneable{
    public int money;

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}
class Student implements Cloneable{
    public String username;
    public String password;
    public Student(String username, String password) {
        this.username = username;
        this.password = password;
    }
    Money money=new Money();
    @Override
    protected Object clone() throws CloneNotSupportedException {
       Student student=(Student)super.clone();
       student.money=(Money)money.clone();
       return student;
    }
    @Override
    public String toString() {
        return "Student{" +
                "username='" + username + '\'' +
                ", password='" + password + '\'' +
                '}';
    }
}
public class HelloWorld{
    public static void main(String[] args) throws CloneNotSupportedException {
        Student student1=new Student("李佳伟","12503487");
        Student student2= (Student) student1.clone();
        student1.money.money=999;
        System.out.println(student1.money.money);//999
        System.out.println(student2.money.money);//0
    }
}

拷贝简单类型是深拷贝,拷贝引用类型是浅拷贝,我们要想实现深拷贝,就要拷贝一下这个对象,只是拷贝地址值,是不能实现真正意义上的深拷贝的

1)咱们的数组的四种拷贝方法都是浅拷贝,因为我们大部分情况下是拷贝地址的,而现在是拷贝数据

2)深拷贝:如果我们拷贝完成之后,我们通过一个引用去访问去修改拷贝完成之后的值,是不会修改原来的值

3)浅拷贝:相反拷贝通过引用来进行修改,原来的值发生变化,就叫做浅拷贝

如果问数组的拷贝是什么拷贝?先问问到底是再拷贝什么???

static关键字修饰的成员是属于类的,而abstract一般是用来修饰普通方法目的是为了让子类继承后重写该方法,而static修饰的方法是不存在继承重写的

创建对象的方式:

  1. new关键字
  2. 反射
  3. Clone方法
  4. 反序列化使用ObjectMapper
;