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}

posted on   乐之者v  阅读(1260)  评论(0编辑  收藏  举报

编辑推荐:
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
阅读排行:
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 记一次.NET内存居高不下排查解决与启示
· DeepSeek 开源周回顾「GitHub 热点速览」
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
历史上的今天:
2018-07-05 如何学习java?
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

导航

统计

点击右上角即可分享
微信分享提示