当前位置: 首页 > news >正文

JavaScript 对象复制:浅拷贝与深拷贝

JavaScript 对象复制:浅拷贝与深拷贝使用说明

在 JavaScript 中,对象复制分为 浅拷贝 和 深拷贝,两者的核心区别在于是否递归复制嵌套的引用类型属性。以下是详细说明和示例:


一、浅拷贝(Shallow Copy)

特点:仅复制对象的第一层属性,若属性是引用类型(如对象、数组),则拷贝的是引用地址。修改嵌套的引用属性会影响原对象。

常用方法:
  1. Object.assign()

    javascript

    const obj = { a: 1, b: { c: 2 } };
    const shallowCopy = Object.assign({}, obj);
    shallowCopy.b.c = 99; // 原对象的 obj.b.c 也会变为 9
    或者
    const shallowCopy = {};
    Object.assign(shallowCopy, obj);也可以复制 reactive 或 ref 创建的响应式对象
    const refObj = ref({});
    Object.assign(refObj.value, proxyObj);
    
  2. 展开运算符 ...

    javascript

    const obj = { a: 1, b: { c: 2 } };
    const shallowCopy = { ...obj };
  3. 数组浅拷贝

    javascript

    const arr = [1, 2, { d: 3 }];
    const newArr = arr.slice(); // 或 [...arr], Array.from(arr)
适用场景:
  • 对象属性无嵌套引用类型。

  • 需要快速复制且性能敏感的场景。


二、深拷贝(Deep Copy)

特点:完全复制对象及其所有嵌套属性,新旧对象完全独立,互不影响。

常用方法:
  1. JSON.parse(JSON.stringify())

    javascript

    const obj = { a: 1, b: { c: 2 }, d: new Date() };
    const deepCopy = JSON.parse(JSON.stringify(obj));如果是 reactive 或 ref 创建的响应式对象,先通过 toRaw 转为 普通对象,再进行拷贝。
    例如:
    const deepCopy = JSON.parse(JSON.stringify(toRaw(proxyObj))); // deepCopy 为普通对象
    const deepCopy = reactive(JSON.parse(JSON.stringify(toRaw(proxyObj)))); // deepCopy 为 reactive 响应式对象 Proxy(Object)
    const deepCopy = ref(JSON.parse(JSON.stringify(toRaw(proxyObj)))); // deepCopy 为ref响应式对象 RefImpl
    

    局限性

    • 忽略 undefined、函数、Symbol。

    • 破坏特殊对象(如 Date 转为字符串,RegExp 转为空对象)。

    • 无法处理循环引用。

  2. structuredClone()(现代浏览器/Node.js 17+)

    javascript

    const obj = { a: 1, b: { c: 2 } };
    const deepCopy = structuredClone(obj); // 支持更多类型,如 Date、Set、Map如果是 reactive 或 ref 创建的响应式对象,先通过 toRaw 转为 普通对象,再进行拷贝。
    例如:
    const deepCopy = structuredClone(toRaw(proxyObj)); // deepCopy 为普通对象
    const deepCopy = reactive(structuredClone(toRaw(proxyObj))); // deepCopy 为 reactive 响应式对象 Proxy(Object)
    const deepCopy = ref(structuredClone(toRaw(proxyObj))); // deepCopy 为ref响应式对象 RefImpl
    
  3. Lodash 的 cloneDeep()

    javascript

    import { cloneDeep } from 'lodash';
    const obj = { a: 1, b: { c: 2 } };
    const deepCopy = cloneDeep(obj); // 处理复杂对象(函数、循环引用等)
  4. 手动递归实现

    javascript

    function deepClone(source, map = new WeakMap()) {if (source === null || typeof source !== 'object') return source;if (map.has(source)) return map.get(source); // 解决循环引用const target = Array.isArray(source) ? [] : {};map.set(source, target);for (const key in source) {if (source.hasOwnProperty(key)) {target[key] = deepClone(source[key], map);}}return target;
    }
适用场景:
  • 对象包含多层嵌套引用。

  • 需要完全独立副本(如状态管理、撤销操作)。


三、如何选择?

方法类型优点缺点
Object.assign浅拷贝简单快速无法处理嵌套引用
JSON 序列化深拷贝原生支持丢失特殊类型,无法处理循环引用
structuredClone深拷贝支持更多类型兼容性要求(IE 不支持)
Lodash cloneDeep深拷贝功能全面需引入第三方库

四、注意事项

  1. 循环引用:手动实现或第三方库需处理对象间的循环引用。

  2. 特殊类型:根据数据类型选择方法(如 structuredClone 支持 Date、Map 等)。

  3. 性能:深拷贝对大型对象有较高开销,优先考虑浅拷贝。


总结

  • 浅拷贝:用 Object.assign 或 ...,适合简单对象。

  • 深拷贝:优先使用 structuredClone 或 Lodash,复杂场景手动实现需谨慎。

通过合理选择拷贝方式,可避免数据意外修改,提升代码健壮性。

相关文章:

  • DBA工作常见问题整理
  • Vue 3 reactive 和 ref 区别及 失去响应性问题
  • Chromium 134 编译指南 macOS篇:获取源代码(四)
  • LeetCode hot 100—括号生成
  • Hyperf (Swoole)的多进程 + 单线程协程、Gin (Go)Go的单进程 + 多 goroutine 解说
  • 深入剖析 ORM:原理、优缺点、场景及多语言框架示例
  • 消除异步的传染性(代数效应)
  • ARINC818-1协议
  • 网易游戏 x Apache Doris:湖仓一体架构演进之路
  • 鸿蒙系统开发中路由使用详解
  • 【并行分布计算】Hadoop伪分布搭建
  • 【并行分布计算】Hadoop单机分布搭建
  • ubuntu docker 创建镜像 报错 dial tcp xxxx read udp xxxx i/o timeout 还有 Forbidden
  • 秘密任务 2.0:如何利用 WebSockets + DTOs 设计实时操作
  • Redis面试——常用命令
  • 【KWDB 创作者计划】_上位机知识篇---SDK
  • Python入门安装和语法基础
  • Ubuntu多用户VNC远程桌面环境搭建:从零开始的完整指南
  • java 设计模式之代理模式
  • [形象解析] ptmalloc、tcmalloc与jemalloc对比
  • 习近平向2025年上海合作组织减贫和可持续发展论坛致贺信
  • 外汇局:4月下旬外资投资境内股票转为净买入
  • 泽连斯基与美国副总统及国务卿会谈,讨论伊斯坦布尔谈判等问题
  • 中国田径巡回赛西安站完赛:男子跳远石雨豪夺冠
  • 打造信息消费新场景、新体验,上海信息消费节开幕
  • 蒲慕明院士:未来数十年不是AI取代人,而是会用AI的人取代不会用的