6.2.3 对象克隆java
Cloneable 接口,这个接口指示一个类提供了一个安全的clone方法。(稍做了解)安全
为一个对象引用的变量创建副本时,原变量和副本都是同一个对象的引用,任何一个变量改变都会影响另外一个变量。ide
Employee original =new Employee("John Public",50000);
Employee copy=original;
copy.raiseSalary(10);//(加工资的一个方法)this
若是但愿copy是一个新对象,它的初始状态与original相同,可是之后它们会有本身不一样的状态,这种状况下就可使用clone方法。spa
Employee copy =original.clone();
copy.raiseSalary(10);(original没有改变)code
对于一个类,须要肯定:对象
1)默认的clone方法是否知足要求;blog
2)是否能够在可变的子对象上调用clone来修补默认的clone方法;接口
3)是否不应使用clone。(默认选项)get
若是选择了第一项或第二项,类必须:
1)实现Cloneable接口;
2)从新定义clone方法,并指定public访问修饰符。
Cloneable接口是java提供的一组标记接口(tagging interface)之一。或者称之为标记接口(marker interface)。接口的一般用途是确保一个类实现一个或一组特定的方法。标记接口不包含任何方法,他惟一的做用就是容许在类型查询中使用instanceof
即便clone的默认(浅拷贝)实现可以知足要求,仍是须要实现Cloneable接口,将clone从新定义为public,在调用super.clone()。下面给出一个例子:
class Employee implements Cloneable { public Employee clone() throws CloneNotSupportedException { return (Employee) super.clone(); } }
要创建深拷贝,(克隆对象中可变的实例域)
class Employee implements Cloneable { public Employee clone() throws CloneNotSupportedException { Employee cloned=(Employee) super.clone(); cloned.hireDay = (Date) hireDay.clone(); return cloned; } }
下列程序克隆了Employee类的一个实例,而后调用两个更改器方法。raiseSalary方法会改变salary域的值,而setHireDay方法改变hireDay域的状态,这两个更改器方法都不会影响原来的对象,由于clone创建了一个深拷贝。
1 package cc.openhome; 2 import java.util.Date; 3 import java.util.GregorianCalendar; 4 public class CloneTest { 5 public static void main(String[] args) { 6 try { 7 Employee original =new Employee("mama", 80000); 8 original.setHireDay(2018, 3, 19); 9 Employee copy=original.clone(); 10 copy.raiseSalary(10); 11 copy.setHireDay(2008, 8, 8); 12 System.out.println("original="+original); 13 System.out.println("copy="+copy); 14 } catch (CloneNotSupportedException e) { 15 e.printStackTrace(); 16 } 17 } 18 } 19 class Employee implements Cloneable 20 { 21 private String name; 22 private double salary; 23 private Date hireDay; 24 public Employee(String name,double salary) 25 { 26 this.name=name; 27 this.salary=salary; 28 hireDay=new Date(); 29 } 30 public Employee clone() throws CloneNotSupportedException 31 { 32 Employee cloned=(Employee) super.clone(); 33 cloned.hireDay = (Date) hireDay.clone(); 34 return cloned; 35 } 36 public void setHireDay(int year,int month,int day) 37 { 38 Date newHireDay =new GregorianCalendar(year, month-1, day).getTime(); 39 hireDay.setTime(newHireDay.getTime()); 40 } 41 public void raiseSalary(double byPercent) 42 { 43 double raise =salary*byPercent/100; 44 salary+=raise; 45 } 46 public String toString() 47 { 48 return "Employee[name="+name+",salary="+salary+",hireDay="+hireDay+"]"; 49 } 50 }


original=Employee[name=mama,salary=80000.0,hireDay=Mon Mar 19 00:00:00 CST 2018] copy=Employee[name=mama,salary=88000.0,hireDay=Fri Aug 08 00:00:00 CST 2008] 成功构建 (总时间: 0 秒)