java中的clone()方法的研究---(8)如何编写正确的clone()方法:数组类型

来源:互联网 发布:百度云管家 mac 编辑:程序博客网 时间:2024/06/05 17:58

在自定义类Person中,会有String, Integer 类型。我们总结的是,针对这些类型的字段,在clone()方法里面,不需要有特别的处理。

但是!!!!!!!

String, Integer类型的数组,在clone()方法里面,是需要特别处理的!!!!


下面来看看我的测试流程。

先在自定义类Person中,添加 String [] , int [] 和 Integer [] 两个属性。


package tt.vo;import java.sql.Timestamp;import java.util.Arrays;import java.util.Date;public class Person implements Cloneable {// 基本数据类型private int age;// Wrapper Class类型private Integer height;// String 类型private String name;// StringBufferprivate StringBuffer address1;// StringBuilderprivate StringBuilder address2;// Dateprivate Date date;// Timestampprivate Timestamp timestamp;// int arrayprivate int[] intArray;// Integer Arrayprivate Integer[] integerArray;// String Arrayprivate String[] stringArray;@Overridepublic Person clone() throws CloneNotSupportedException {Person p = (Person) super.clone();// StringBuffer,StringBuilder 没有实现clone方法// 只能用 new, 去完成克隆动作p.address1 = new StringBuffer(this.address1);p.address2 = new StringBuilder(this.address2);p.date = (Date) this.date.clone();p.timestamp = (Timestamp) this.timestamp.clone();// 这里我先给注释掉// p.intArray = this.intArray.clone();// p.integerArray = this.integerArray.clone();// p.stringArray = this.stringArray.clone();return p;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public Integer getHeight() {return height;}public void setHeight(Integer height) {this.height = height;}public String getName() {return name;}public void setName(String name) {this.name = name;}public StringBuffer getAddress1() {return address1;}public void setAddress1(StringBuffer address1) {this.address1 = address1;}public StringBuilder getAddress2() {return address2;}public void setAddress2(StringBuilder address2) {this.address2 = address2;}@Overridepublic String toString() {return "Person [age=" + age + ", height=" + height + ", name=" + name+ ", address1=" + address1 + ", address2=" + address2+ ", date=" + date + ", timestamp=" + timestamp + ", intArray="+ Arrays.toString(intArray) + ", integerArray="+ Arrays.toString(integerArray) + ", stringArray="+ Arrays.toString(stringArray) + "]";}public Date getDate() {return date;}public void setDate(Date date) {this.date = date;}public Timestamp getTimestamp() {return timestamp;}public void setTimestamp(Timestamp timestamp) {this.timestamp = timestamp;}public Integer[] getIntegerArray() {return integerArray;}public void setIntegerArray(Integer[] integerArray) {this.integerArray = integerArray;}public String[] getStringArray() {return stringArray;}public void setStringArray(String[] stringArray) {this.stringArray = stringArray;}public int[] getIntArray() {return intArray;}public void setIntArray(int[] intArray) {this.intArray = intArray;}}



---------------------------------------------------------------下面开始测试了------------------------------------------------------------------

测试类TestMain:


package tt;import java.sql.Timestamp;import java.util.Date;import tt.vo.Person;public class TestMain {public static void main(String[] args) throws CloneNotSupportedException {// initialize Person pPerson p = new Person();// int p.setAge(18);// Integerp.setHeight(162);// Stringp.setName("郭美美");// StringBufferp.setAddress1(new StringBuffer("中国北京"));// StringBuilderp.setAddress2(new StringBuilder("美国纽约"));// Datep.setDate(new Date());// Timestampp.setTimestamp(new Timestamp(123456789l));// int arrayp.setIntArray(new int [] {10,11,12,13});// Integer arrayp.setIntegerArray(new Integer [] {1,2,3,4});// String arrayp.setStringArray(new String [] {"a","b","c","d"});System.out.println("p:" + p);Person pclone = p.clone();System.out.println("pclone:" + pclone);System.out.println("-------------after set-------------------");p.setAge(180);p.setHeight(190);p.setName("郭美美不美");pclone.getAddress1().append("!!!");pclone.getAddress2().append("......");p.getDate().setYear(2000);p.getTimestamp().setYear(3000);p.getIntArray()[3] = 1000;p.getIntegerArray()[3] = 4000;p.getStringArray()[3] = "dddd";System.out.println("p:" + p);System.out.println("pclone:" + pclone);}}



debug截图:




通过debug可以发现:p.intArray, p.integerArray p.stringArray 和 克隆出来的pclone.intArray, pclone.integerArray  pclone.stringArray的id完全相同。包括数组本身的id(36 39 43)和数组元素的id

当如下代码被执行之后,我们再来看看debug的截图

p.getIntArray()[3] = 1000;p.getIntegerArray()[3] = 4000;p.getStringArray()[3] = "dddd";




通过debug可以发现:

  • 对p.getIntegerArray()[3] 的修改,同时也影响了 pclone.getIntegerArray()[3]
  • 在修改p.getIntegerArray()[3]之后,  p.getIntegerArray()[3]  和 pclone.getIntegerArray()[3] 同时指向了一个新的内存地址 (id=383)
  • p.getStringArray()[3] 也是一样
  • p.getIntArray()[3] 因为是int 数组,所以数组元素没有id。但是效果是一样的,对p.getIntArray()[3]的修改,同时也修改了pclone.getIntArray()[3]
  • 好吧,这显然不是我们想要的结果。


解决办法如下,修改clone()方法

@Overridepublic Person clone() throws CloneNotSupportedException {Person p = (Person) super.clone();// StringBuffer,StringBuilder 没有实现clone方法// 只能用 new, 去完成克隆动作p.address1 = new StringBuffer(this.address1);p.address2 = new StringBuilder(this.address2);p.date = (Date) this.date.clone();p.timestamp = (Timestamp) this.timestamp.clone();p.intArray = this.intArray.clone();p.integerArray = this.integerArray.clone();p.stringArray = this.stringArray.clone();return p;}




哈哈,你会发现无论什么类型的数组,其实都自带clone()方法。不知道为什么String类型没有clone()望大师们赐教!!!害羞

修改了clone()方法之后,我们再debug看看:





通过debug可以发现:

  • 从id上看p.integerArray(id=50) ,pclone.integerArray(id=39) 本身的id 不同 
  • 但是其数组元素的id都是相同的!!!!!!!!!!!!!!!!!!!!
  • p.stringArray 与 pclone.stringArray之间,   p.intArray 与pclone.intArray之间, 也是相同的现象。
  • 那么这样id小小的不同点,会有如何质的变化呢?我们接着来看


测试方法还是一样:

p.getIntArray()[3] = 1000;p.getIntegerArray()[3] = 4000;p.getStringArray()[3] = "dddd";


来看看debug 截图:


通过debug可以发现:
  • 在修改p.getIntegerArray()[3]之后,  只是p.getIntegerArray()[3]  指向了一个新的内存地址 (id=172)。pclone.getIntegerArray()[3]不变
  • 修改了p.getStringArray()[3] 和p.getIntArray()[3]  的现象也是一样
  • 好吧,这就是我们想要的结果。

来看看控制台:

-------------after set-------------------
p:Person [integerArray=[1, 2, 3, 4000], stringArray=[a, b, c, dddd]]
pclone:Person [integerArray=[1, 2, 3, 4], stringArray=[a, b, c, d]]



1 0
原创粉丝点击