在java开发中,有时我们需要复制对象,并且确保修改复制得到的对象不会影响原来的对象。
于是,有些人可能会写出类似以下的代码:
public class CloneTest {
public static void main(String[] args) {
JSONObject paramJson= new JSONObject();
paramJson.put("age",26);
paramJson.put("name","lin");
JSONObject cloneJson= paramJson;
cloneJson.put("test",27);
System.out.println("paramJson:"+paramJson.toString());
System.out.println("cloneJson:"+cloneJson.toString());
}
}
运行结果为:
paramJson:{"age":27,"name":"lin"}
cloneJson:{"age":27,"name":"lin"}
发现赋值得到的对象改变时,原来的对象也随之改变。
在值传递与引用传递中已经讲到,
引用类型传参或赋值,新的对象改变时,原来的对象也会改变。
那么,如何确保修改复制得到的新对象不会影响原来的对象呢?
深拷背和浅拷背
复制对象,修改新对象不会影响原来的对象。这其实就是深拷背。
而修改复制得到的新对象会影响原来的对象,则是浅拷背。
从值和引用的角度理解,概念如下:
1.浅拷贝: 将原对象或原数组的引用直接赋给新对象,新数组,新对象/数组只是原对象的一个引用
2.深拷贝: 创建一个新的对象和数组,将原对象的各项属性的“值”(数组的所有元素)拷贝过来,是“值”而不是“引用”
在java的超类Object中,有一个clone()方法,默认的是浅拷背。
protected Object clone()
throws CloneNotSupportedException
那么,如何实现深拷背呢?
深拷背的实现
有几个工具类可以使用深拷背。
1.BeanUtils。
注意,最好使用Spring的BeanUtils,不要用Apache-Commons的BeanUtils。
阿里巴巴java规范手册中提到,Apache BeanUtils性能较差,可以使用 Spring BeanUtils 或者 Cglib BeanCopier 来代替。
public class DeepCloneTest {
public static void main(String[] args) throws InvocationTargetException, IllegalAccessException {
JSONObject paramJson= new JSONObject();
paramJson.put("age",26);
paramJson.put("name","lin");
JSONObject cloneJson= new JSONObject();
BeanUtils.copyProperties(cloneJson,paramJson);
cloneJson.put("test",27);
System.out.println("paramJson:"+paramJson.toString());
System.out.println("cloneJson:"+cloneJson.toString());
}
}
运行结果:
paramJson:{"name":"lin","age":26}
cloneJson:{"test":27}
2.apache-commons包的SerializationUtils。**SerializationUtils要求对象必须实现序列化接口 **
public class DeepCloneTest {
public static void main(String[] args) {
JSONObject paramJson= new JSONObject();
paramJson.put("age",26);
paramJson.put("name","lin");
JSONObject cloneJson=SerializationUtils.clone(paramJson);
cloneJson.put("age",27);
System.out.println("paramJson:"+paramJson.toString());
System.out.println("cloneJson:"+cloneJson.toString());
}
}
运行结果:
paramJson:{"name":"long","age":26}
cloneJson:{"name":"long","age":27}