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 }
![](http://static.javashuo.com/static/loading.gif)
![](http://static.javashuo.com/static/loading.gif)
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 秒)