news 2026/6/9 22:04:00

深拷贝和浅拷贝的区别,以及BeanUtils为什么是浅拷贝

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
深拷贝和浅拷贝的区别,以及BeanUtils为什么是浅拷贝

深拷贝(Deep Copy)和浅拷贝(Shallow Copy)是编程中复制对象时的两种不同方式,核心区别在于是否递归复制对象内部的引用类型成员


一、浅拷贝(Shallow Copy)

✅ 定义:
  • 创建一个新对象
  • 基本数据类型(如intStringboolean)的值会被直接复制(值拷贝)。
  • 引用类型(如对象、数组、List)只复制引用地址,不复制实际对象。
🧠 内存效果:
原对象 A ────┬──→ 基本字段:10(独立) └──→ 引用字段:Addr@123(指向堆中地址对象) 浅拷贝 B ───┬──→ 基本字段:10(独立副本) └──→ 引用字段:Addr@123(和 A 共享同一个地址对象!)
⚠️ 特点:
  • 修改基本类型字段:互不影响。
  • 修改引用类型字段的内容(如address.city = "上海"):会影响原对象!
💡 示例(Java):
class Address { String city; } class Person { String name; Address address; } // 浅拷贝(使用 clone() 默认行为) Person p1 = new Person(); p1.name = "张三"; p1.address = new Address(); p1.address.city = "北京"; Person p2 = (Person) p1.clone(); // 浅拷贝 p2.address.city = "上海"; // 修改引用对象 System.out.println(p1.address.city); // 输出 "上海" → 被影响了!

二、深拷贝(Deep Copy)

✅ 定义:
  • 创建一个全新的对象
  • 递归复制所有层级:不仅复制对象本身,还复制它引用的所有对象(子对象、孙子对象……)。
  • 新旧对象完全独立,没有任何共享引用。
🧠 内存效果:
原对象 A ───→ Addr@123(city="北京") 深拷贝 B ───→ Addr@456(city="北京")← 全新对象!
✅ 特点:
  • 修改任何字段(包括嵌套对象):都不会影响原对象
💡 示例(Java 手动实现):
class Address implements Cloneable { String city; public Address deepCopy() { Address a = new Address(); a.city = this.city; return a; } } class Person { String name; Address address; public Person deepCopy() { Person p = new Person(); p.name = this.name; p.address = this.address.deepCopy(); // 关键:也拷贝子对象 return p; } } Person p1 = new Person(); p1.address = new Address(); p1.address.city = "北京"; Person p2 = p1.deepCopy(); p2.address.city = "上海"; System.out.println(p1.address.city); // 输出 "北京" → 不受影响!

三、对比总结

特性浅拷贝深拷贝
基本类型字段复制值(独立)复制值(独立)
引用类型字段复制引用(共享对象)复制对象(全新实例)
内存占用
性能慢(需递归)
修改影响可能互相影响完全独立
实现难度简单(如clone()复杂(需递归或序列化)

四、常见语言中的实现方式

JavaScript
  • 浅拷贝:{...obj},Object.assign(),arr.slice()
  • 深拷贝:
    • JSON.parse(JSON.stringify(obj))(有局限)
    • structuredClone(obj)(现代浏览器)
    • _.cloneDeep()(Lodash 库)
Python
  • 浅拷贝:copy.copy(obj)
  • 深拷贝:copy.deepcopy(obj)
Java
  • 浅拷贝:Object.clone()(默认)
  • 深拷贝:
    • 重写clone()并递归克隆引用字段
    • 序列化/反序列化(要求实现Serializable

五、一句话记住

浅拷贝:只复制“外壳”,内部对象共享。
深拷贝:复制“整个家族”,完全独立。

当你需要完全隔离的数据副本(如导出、快照、撤销操作),用深拷贝;
如果只是临时读取或性能敏感,且不会修改嵌套对象,可用浅拷贝。

BeanUtils.copyProperties()不是深拷贝,而是典型的浅拷贝(Shallow Copy)


✅ 官方结论

无论是Spring Framework 的org.springframework.beans.BeanUtils.copyProperties
还是Apache Commons BeanUtils 的org.apache.commons.beanutils.BeanUtils.copyProperties
它们都只执行浅拷贝


🔍 什么是“浅拷贝”在此处的含义?

  • 对于基本数据类型(如int,String,Boolean):
    值会被复制到目标对象 → 修改互不影响 ✅
  • 对于引用类型(如Address,List, 自定义对象):
    只复制引用地址,不创建新对象 → 源对象和目标对象共享同一个子对象实例

这意味着:

如果你修改了目标对象中某个嵌套对象的属性,源对象也会被影响


🧪 示例说明(使用 Spring BeanUtils)

// 定义类 class Address { private String city; // getter/setter } class Person { private String name; private Address address; // getter/setter }
// 使用 copyProperties Person source = new Person(); source.setName("张三"); source.setAddress(new Address()); source.getAddress().setCity("北京"); Person target = new Person(); BeanUtils.copyProperties(source, target); // 浅拷贝! // 修改 target 的嵌套对象 target.getAddress().setCity("上海"); // 结果: System.out.println(source.getAddress().getCity()); // 输出 "上海"! System.out.println(target.getAddress().getCity()); // 输出 "上海"

💥 两个对象的address字段指向同一个Address实例,因此修改一个会影响另一个。


⚠️ 为什么不是深拷贝?

BeanUtils.copyProperties的工作原理是:

  1. 通过反射获取源对象的getter 方法
  2. 调用目标对象的setter 方法
  3. 直接赋值返回的对象引用

不会递归地为每个引用类型字段创建新实例,因此无法实现深拷贝 。


✅ 如何实现真正的深拷贝?

如果你需要深拷贝,可以考虑以下方案:

方法说明缺点
手动逐层复制在 setter 中 new 子对象并复制属性代码冗长,维护成本高
序列化/反序列化使用ObjectOutputStream或 FastJSON/Jackson要求所有类实现Serializable,性能较差
第三方库如 Dozer、ModelMapper、MapStruct需引入依赖,学习成本
JSON 中转(慎用)JSON.parseObject(JSON.toJSONString(obj), Clazz)可能丢失类型信息(如 Date 变时间戳),且有性能开销

示例(Jackson 深拷贝):

ObjectMapper mapper = new ObjectMapper(); String json = mapper.writeValueAsString(source); Target target = mapper.readValue(json, Target.class); // 真正的深拷贝

📌 总结

问题回答
BeanUtils.copyProperties是深拷贝吗?不是,是浅拷贝
能否用于含嵌套对象的 DTO 转换?⚠️可以,但要确保后续不修改嵌套对象,否则会污染源数据
修改目标对象会影响源对象吗?如果修改的是引用类型的内部状态,会!
推荐在什么场景使用?✅ 仅包含基本类型或不可变对象(如String)的 POJO 复制

💡最佳实践
当你的 DTO/Entity没有嵌套对象,或嵌套对象不会被修改时,BeanUtils.copyProperties是安全高效的;
一旦涉及可变的复杂对象图,请改用深拷贝方案 。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 2:19:29

[STM32C0] 【STM32C092RC 测评】SPI Loopback测试

本次测试聚焦于SPI(串行外设接口),一种广泛应用于嵌入式系统及电子设备的高速、全双工、同步通信总线。为全面且有效地评估SPI接口功能,我们采用Loopback(回环)测试方法。SPI通信架构核心包含四个关键信号引…

作者头像 李华
网站建设 2026/6/10 19:15:26

电商GIF主图压缩教程,专业GIF压缩工具优化攻略

做自媒体配图、电商主图或课件动图时,总遇到选不好GIF 压缩工具、压缩后画质模糊,或是压缩后体积仍不达标无法上传的问题,换多款工具反复调试,既浪费时间又达不到预期效果。不同平台对 GIF 有明确规格要求:微信动图建议…

作者头像 李华
网站建设 2026/6/10 20:02:01

[Windows] 视频伪装大师VideoCamouflageMaster v1.0.0

[Windows] 视频伪装大师VideoCamouflageMaster v1.0.0 链接:https://pan.xunlei.com/s/VOjZqRSchB2LHT_us9a3U5_EA1?pwd5c3g# 本软件是一款专注于个人隐私保护的桌面多媒体工具。它不仅能够对用户的私密视频和图片进行高强度加密,还具备独特的“文件伪…

作者头像 李华
网站建设 2026/6/10 17:14:33

【大数据毕设源码分享】基于Python的农业大数据管理系统的设计与实现(程序+文档+代码讲解+一条龙定制)

博主介绍:✌️码农一枚 ,专注于大学生项目实战开发、讲解和毕业🚢文撰写修改等。全栈领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战 ✌️技术范围:&am…

作者头像 李华