百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

浅拷贝与深拷贝全面解析及实战(什么是浅拷贝什么是深拷贝)

wxin55 2024-11-17 02:43 8 浏览 0 评论

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

浅拷贝:表面级复制

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

实现方法:

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()
  • const arr = [1, 2, 3];
    const newArr = [].concat(arr);
    console.log(newArry); // 输出 [1, 2, 3]
    • Array.slice(0)
        const arr = [1, 2, 3];
        const newArr = original.slice(0);
        console.log(newArr); // 输出 [1, 2, 3]
    • Array.toReverse().reverse()
        const arr = [1, 2, 3];
        const newArr = arr.toReversed().reverse();
        console.log(newArr); // 输出 [1, 2, 3]
    • 扩展运算符 ...args
        const arr = [1, 2];
        const newArr = [...arr];
        console.log(newArr); 

    手写浅拷贝示例

    • 实现原理:forin循环遍历对象中的所有属性借助hasOwnProperty()方法,判断属性是否为对象显式属性
    • 实现代码:
    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))(有局限性)
         const original = { a: 1, b: { c: 2 } };
         const copy = JSON.parse(JSON.stringify(original));
         console.log(copy); // 输出 { a: 1, b: { c: 2 } }
    • structuredClone(obj)(较新,但仍有限制)
        const original = { a: 1, b: { c: 2 } };
        const copy = structuredClone(original);
        console.log(copy); // 输出 { a: 1, b: { c: 2 } }
    • 自定义递归函数实现

    手写深拷贝示例

    • 实现原理:
    1. forin循环遍历对象中的所有属性
    2. 借助hasOwnProperty()方法,判断属性是否为对象显式属性
    3. 判断属性值类型,递归调用深拷贝函数
    • 实现代码:
    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对象的理解,也能在特定场景下提供灵活的解决方案。


    文章转自:https://juejin.cn/post/7379151898567622696

    相关推荐

    ES6中 Promise的使用场景?(es6promise用法例子)

    一、介绍Promise,译为承诺,是异步编程的一种解决方案,比传统的解决方案(回调函数)更加合理和更加强大在以往我们如果处理多层异步操作,我们往往会像下面那样编写我们的代码doSomething(f...

    JavaScript 对 Promise 并发的处理方法

    Promise对象代表一个未来的值,它有三种状态:pending待定,这是Promise的初始状态,它可能成功,也可能失败,前途未卜fulfilled已完成,这是一种成功的状态,此时可以获取...

    Promise的九大方法(promise的实例方法)

    1、promise.resolv静态方法Promise.resolve(value)可以认为是newPromise方法的语法糖,比如Promise.resolve(42)可以认为是以下代码的语...

    360前端一面~面试题解析(360前端开发面试题)

    1.组件库按需加载怎么做的,具体打包配了什么-按需加载实现:借助打包工具(如Webpack的require.context或ES模块动态导入),在使用组件时才引入对应的代码。例如在V...

    前端面试-Promise 的 finally 怎么实现的?如何在工作中使用?

    Promise的finally方法是一个非常有用的工具,它无论Promise是成功(fulfilled)还是失败(rejected)都会执行,且不改变Promise的最终结果。它的实现原...

    最简单手写Promise,30行代码理解Promise核心原理和发布订阅模式

    看了全网手写Promise的,大部分对于新手还是比较难理解的,其中几个比较难的点:状态还未改变时通过发布订阅模式去收集事件实例化的时候通过调用构造函数里传出来的方法去修改类里面的状态,这个叫Re...

    前端分享-Promise可以中途取消啦(promise可以取消吗)

    传统Promise就像一台需要手动组装的设备,每次使用都要重新接线。而Promise.withResolvers的出现,相当于给开发者发了一个智能遥控器,可以随时随地控制异步操作。它解决了三大...

    手写 Promise(手写输入法 中文)

    前言都2020年了,Promise大家肯定都在用了,但是估计很多人对其原理还是一知半解,今天就让我们一起实现一个符合PromiseA+规范的Promise。附PromiseA+规范地址...

    什么是 Promise.allSettled()!新手老手都要会?

    Promise.allSettled()方法返回一个在所有给定的promise都已经fulfilled或rejected后的promise,并带有一个对象数组,每个对象表示对应的pr...

    前端面试-关于Promise解析与高频面试题示范

    Promise是啥,直接上图:Promise就是处理异步函数的API,它可以包裹一个异步函数,在异步函数完成时抛出完成状态,让代码结束远古时无限回掉的窘境。配合async/await语法糖,可...

    宇宙厂:为什么前端离不开 Promise.withResolvers() ?

    大家好,很高兴又见面了,我是"高级前端进阶",由我带着大家一起关注前端前沿、深入前端底层技术,大家一起进步,也欢迎大家关注、点赞、收藏、转发。1.为什么需要Promise.with...

    Promise 新增了一个超实用的 API!

    在JavaScript的世界里,Promise一直是处理异步操作的神器。而现在,随着ES2025的发布,Promise又迎来了一个超实用的新成员——Promise.try()!这个新方法简...

    一次搞懂 Promise 异步处理(promise 异步顺序执行)

    PromisePromise就像这个词的表面意识一样,表示一种承诺、许诺,会在后面给出一个结果,成功或者失败。现在已经成为了主流的异步编程的操作方式,写进了标准里面。状态Promise有且仅有...

    Promise 核心机制详解(promise机制的实现原理)

    一、Promise的核心状态机Promise本质上是一个状态机,其行为由内部状态严格管控。每个Promise实例在创建时处于Pending(等待)状态,此时异步操作尚未完成。当异步操作成功...

    javascript——Promise(js实现promise)

    1.PromiseES6开始支持,Promise对象用于一个异步操作的最终完成(包括成功和失败)及结果值的表示。简单说就是处理异步请求的。之所以叫Promise,就是我承诺,如果成功则怎么处理,失败怎...

    取消回复欢迎 发表评论: