浅拷轻描生新贝,深复细琢铸宝胚

在JavaScript学习中,拷贝是很重要的一个知识点。拷贝主要分为浅拷贝(Shallow Copy)和深拷贝(Deep Copy)。本文将阐述两者的概念,还将通过手写示例深入探讨如何实现这两种拷贝方式,以及它们在实际应用中的考量。

浅拷贝:表面级复制

浅拷贝只复制对象的第一层属性,对于嵌套对象或数组,仅复制它们的引用,导致原对象和拷贝对象在这些部分仍然共享数据。

实现方法:

js 复制代码
let obj = {
    a: 1, 
    b: [1,2,3]
}
  • Object.create(obj)

    • let obj2 = Object.create(obj);
  • Object.assign({}, obj)

    • let obj2 = Object.assign({}, obj);
  • Array.concat()

    js 复制代码
    const arr = [1, 2, 3];
    const newArr = [].concat(arr);
    console.log(newArry); // 输出 [1, 2, 3]
  • Array.slice(0)

js 复制代码
    const arr = [1, 2, 3];
    const newArr = original.slice(0);
    console.log(newArr); // 输出 [1, 2, 3]
  • Array.toReverse().reverse()
js 复制代码
    const arr = [1, 2, 3];
    const newArr = arr.toReversed().reverse();
    console.log(newArr); // 输出 [1, 2, 3]
  • 扩展运算符 ...args
js 复制代码
    const arr = [1, 2];
    const newArr = [...arr];
    console.log(newArr); 

手写浅拷贝示例

  • 实现原理:
    1. forin循环遍历对象中的所有属性
    2. 借助hasOwnProperty()方法,判断属性是否为对象显式属性
  • 实现代码:
javascript 复制代码
function shallowCopy(obj) {
    let newObj = Array.isArray(obj) ? [] : {};
    for (let key in obj) {
        if (obj.hasOwnProperty(key)) {
            newObj[key] = obj[key];
        }
    }
    return newObj;
}

特点

  • 执行速度快,占用资源少。
  • 适用于简单对象或不需要完全独立拷贝的场景。
  • 不适用于包含嵌套对象或数组的复杂结构,因为更改嵌套数据会影响原对象。

深拷贝:彻底复制

深拷贝会递归地复制对象的所有层次,包括嵌套的数组和对象,从而确保原对象和拷贝对象完全独立,互不影响。

实现方法:

  • JSON.parse(JSON.stringify(obj))(有局限性)
js 复制代码
     const original = { a: 1, b: { c: 2 } };
     const copy = JSON.parse(JSON.stringify(original));
     console.log(copy); // 输出 { a: 1, b: { c: 2 } }
  • structuredClone(obj)(较新,但仍有限制)
js 复制代码
    const original = { a: 1, b: { c: 2 } };
    const copy = structuredClone(original);
    console.log(copy); // 输出 { a: 1, b: { c: 2 } }
  • 自定义递归函数实现

手写深拷贝示例

  • 实现原理:
  1. forin循环遍历对象中的所有属性
  2. 借助hasOwnProperty()方法,判断属性是否为对象显式属性
  3. 判断属性值类型,递归调用深拷贝函数
  • 实现代码:
javascript 复制代码
function deepCopy(obj) {
    if (!obj || typeof obj !== 'object') {
        return obj;
    }
    let newObj = Array.isArray(obj) ? [] : {};
    for (let key in obj) {
        if (obj.hasOwnProperty(key)) {
            newObj[key] = deepCopy(obj[key]);
        }
    }
    return newObj;
}

特点

  • 创建完全独立的副本,适用于复杂数据结构。
  • 相对耗时和占用更多资源,尤其是面对深度嵌套对象。

实践比较与选择

  • 性能与资源:浅拷贝因为操作简单,执行效率高,适合快速复制;深拷贝则因递归复制所有层级,成本较高,但在需要完全独立数据副本时不可或缺。
  • 应用场景:对于简单的对象复制或状态克隆,浅拷贝足矣;而在需要确保数据完全隔离,避免外部修改影响内部状态时,深拷贝是更好的选择。
  • 注意事项 :使用JSON.stringify()JSON.parse()进行深拷贝虽简便,但存在局限性(如不支持函数、Symbol、循环引用等)。自定义深拷贝时,循环引用的处理是一个常见的挑战。

结论

浅拷贝和深拷贝各有千秋,关键在于根据具体需求选择合适的拷贝策略。了解它们的实现机制,不仅能提升代码的健壮性,还能有效避免潜在的数据篡改问题。通过手写实现深浅拷贝,不仅可以加深对JavaScript对象的理解,也能在特定场景下提供灵活的解决方案。

相关推荐
2501_920931705 小时前
React Native鸿蒙跨平台采用ScrollView的horizontal属性实现横向滚动实现特色游戏轮播和分类导航
javascript·react native·react.js·游戏·ecmascript·harmonyos
东东5167 小时前
智能社区管理系统的设计与实现ssm+vue
前端·javascript·vue.js·毕业设计·毕设
catino7 小时前
图片、文件的预览
前端·javascript
2501_920931708 小时前
React Native鸿蒙跨平台实现推箱子游戏,完成玩家移动与箱子推动,当所有箱子都被推到目标位置时,玩家获胜
javascript·react native·react.js·游戏·ecmascript·harmonyos
AI老李9 小时前
PostCSS完全指南:功能/配置/插件/SourceMap/AST/插件开发/自定义语法
前端·javascript·postcss
方也_arkling9 小时前
Element Plus主题色定制
javascript·sass
2601_9498095910 小时前
flutter_for_openharmony家庭相册app实战+我的Tab实现
java·javascript·flutter
Up九五小庞10 小时前
开源埋点分析平台 ClkLog 本地部署 + Web JS 埋点测试实战--九五小庞
前端·javascript·开源
摘星编程10 小时前
React Native + OpenHarmony:UniversalLink通用链接
javascript·react native·react.js
qq_1777673710 小时前
React Native鸿蒙跨平台数据使用监控应用技术,通过setInterval每5秒更新一次数据使用情况和套餐使用情况,模拟了真实应用中的数据监控场景
开发语言·前端·javascript·react native·react.js·ecmascript·harmonyos