java复制对象之深拷背

在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}
原文地址:https://www.cnblogs.com/expiator/p/11135968.html