疯狂java


您现在的位置: 疯狂软件 >> 新闻资讯 >> 正文

Java的浅拷贝和深拷贝


 

   

  首先用一个小例子来讲解一个引用和拷贝的区别:

  先定义两个基本类型:

  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