首先用一个小例子来讲解一个引用和拷贝的区别:
先定义两个基本类型:
class BirthDayimplements Cloneable{
public int year;
public int month;
public int day;
public BirthDay(int year, int month, int day) {
this.year = year;
this.month = month;
this.day = day;
}
}
class User implements Cloneable{
public String name;
public int age;
public BirthDay birth;
public User(String name, int age, BirthDay birth) {
this.name = name;
this.age = age;
this.birth = birth;
}
public User clone() throws CloneNotSupportedException {
return (User)super.clone();
}
}
User userA = new User("Jack", 20, new BirthDay(1994, 1, 1));
User userB = userA;
userB.age = 25;
System.out.println(userA.age);
输出结果是:25
首先new出一个userA对象,然后将userA赋给userB,那么userA和userB两个对象同时指向同一个内存地址,操作userB中的变量值,会影响userA中的变量值。
那么如何将userA拷贝一份,从而达到更改userB中的变量值,而不会改动原本userA中的变量值呢?很简单,可以使用下面的方法:
User userB = userA.clone();
Clone是如何完成的呢?Object在对某个对象实施Clone时对其是一无所知的,它仅仅是简单地执行域对域的copy,这就是浅拷贝
。
那么对于userA和userB中不是基本类型的birth变量,也只是单纯地执行对域的复制而已,同样指向同一块内存,那么操作userA中的birth变量,同样还是会影响到userB中的birth变量。
那么如何做到深拷贝,使得无论操作userB的各个变量都不会影响到userA中的变量值呢?
只需要在原有的基础上做一点改动,就是在UserA中的所有非基本类型变量的类(如:BirthDay类)实现Cloneable接口,并且重载clone方法即可,当调用clone方法时,系统会将所有变量都克隆一份,这样就可以达到userA和userB两个对象的操作都互不影响了。
注意:
1>clone()方法的保护机制在Object中clone()是被申明为protected的
2>调用clone()方法的对象所属的类(Class)必须implements Clonable接口,否则在调用Clone方法的时候会抛出CloneNotSupportedException。
3>重载clone()方法的时候,需要将clone方法的修饰符更改为public