当前位置: 首页 > 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,复杂场景手动实现需谨慎。

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

http://www.xdnf.cn/news/3601.html

相关文章:

  • 观察者模式与发布订阅模式:解耦与通信的艺术
  • 【网络】IP层的重要知识
  • 【工具】视频翻译、配音、语音克隆于一体的一站式视频多语言转换工具~
  • Redis面试——事务
  • NoSQL 与 NewSQL 全面对比:如何选择适合你的数据库方案?
  • 使用代理IP提取数据的步骤是什么?代理IP如何提高爬虫采集效率?
  • 2000-2017年各省国有经济煤气生产和供应业固定资产投资数据
  • 从基础概念到前沿应用了解机器学习
  • VFlash的自动化和自定义动作
  • 山东大学软件学院创新项目实训开发日志(15)之中医知识问答历史对话查看bug处理后端信息响应成功但前端未获取到
  • 【模块化拆解与多视角信息6】自我评价:人设构建的黄金50字——从无效堆砌到精准狙击的认知升级
  • 影视产业链中的律师角色以及合规风控要点
  • 【React】项目的搭建
  • B端小程序如何突破常规,成为企业获客新利器?
  • win10下github libiec61850库编译调试sntp_example
  • XCZU4CG‑2SFVC784I 赛灵思 FPGA XilinxZynq UltraScale+ MPSoC
  • JVM详解(曼波脑图版)
  • 【WPF】 自定义控件的自定义属性
  • OpenCV颜色变换cvtColor
  • UE中通过AAIController::MoveTo函数巡逻至目标点后没法正常更新巡逻目标点
  • 力扣349 == 两个数组交集的两种解法
  • selenium 实现模拟登录中的滑块验证功能
  • 【文献笔记】LLM-based control code generation using image recognition
  • 详细讲解一下Java中的Enum
  • Missashe考研日记-day22
  • AI与物联网的深度融合:开启智能生活新时代
  • 写书的三驾马车
  • 【Unity】UI点击事件处理器
  • 文件系统 软硬连接
  • 类和对象(下篇)(详解)