这里简单记录一下java中关于浅复制和深复制的知识。很多时候,一个人选择了行走,不是因为欲望,也并非诱惑,他仅仅是听到了自己内心的声音。
java中的复制clone方法
一、java对象的浅复制
一个实现了Cloneable并重写了clone方法的类A,有一个无参构造或有参构造B,通过new关键字产生了一个对象S,再然后通过S.clone()方式产生了一个新的对象T,那么在对象拷贝时构造函数B是不会被执行的
public class Thing implements Cloneable { //定义一个私有变量 private ArrayList<String> arrayList = new ArrayList<String>(); public Thing() { System.out.println("构造函数被执行了......"); } @Override protected Thing clone() { Thing thing = null; try { thing = (Thing) super.clone(); } catch (CloneNotSupportedException e) { e.printStackTrace(); } return thing; } //设置HashMap的值 public void setValue(String value) { this.arrayList.add(value); } //取得arrayList的值 public List<String> getValue() { return this.arrayList; } }
Thing类实现了Cloneable接口,并重写了Object的clone方法,里面有一个ArrayList列表的私有变量。现在我们的测试代码如下:
package com.linux.huhx.learn.clone; /** * @Author: huhx * @Date: 2017-12-26 上午 10:57 */ public class ClientMain { public static void main(String[] args) { // 产生一个对象 Thing thing = new Thing(); thing.setValue("huhx"); // 复制一个对象 Thing cloneThing = thing.clone(); cloneThing.setValue("linux"); System.out.println(thing.getValue()); } }
运行的结果如下:
构造函数被执行了......
[huhx, linux]
从结果我们可以看到:thing.clone()执行并没有再次调用构造函数,复制后的对象和原对象共享私有变量arrayList,也就是说复制对象对arrayList的操作会映射到原对象的arrayList。这样的浅拷贝是有风险的,下面我们介绍一下怎样实现对象的深复制。我们只需要在Thing类中clone方法加上数组的复制代码就可以,clone方法的代码如下:
@Override protected Thing clone() { Thing thing = null; try { thing = (Thing) super.clone(); thing.arrayList = (ArrayList<String>) this.arrayList.clone(); } catch (CloneNotSupportedException e) { e.printStackTrace(); } return thing; }
现在再执行ClientMain的main方法,打印的结果如下:可以看到原对象的arrayList并没有改变。
构造函数被执行了......
[huhx]
需要注意的是:对象的clone与对象内的final关键字是有冲突的,我们修改Thing类的私有变量arrayList的修饰符。
private final ArrayList<String> arrayList = new ArrayList<String>();
通过ide我们可以看到在thing.arrayList = (ArrayList<String>) this.arrayList.clone();代码处会有编译报错。所以要想实现对象的深复制,类的成员变量上不要增加final关键字。
Cannot assign a value to final variable 'arrayList'