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

浅拷贝与深拷贝全面解析及实战(浅拷贝和深拷贝的实现方式)

wptr33 2025-03-20 20:08 25 浏览

在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

    相关推荐

    oracle数据导入导出_oracle数据导入导出工具

    关于oracle的数据导入导出,这个功能的使用场景,一般是换服务环境,把原先的oracle数据导入到另外一台oracle数据库,或者导出备份使用。只不过oracle的导入导出命令不好记忆,稍稍有点复杂...

    继续学习Python中的while true/break语句

    上次讲到if语句的用法,大家在微信公众号问了小编很多问题,那么小编在这几种解决一下,1.else和elif是子模块,不能单独使用2.一个if语句中可以包括很多个elif语句,但结尾只能有一个else解...

    python continue和break的区别_python中break语句和continue语句的区别

    python中循环语句经常会使用continue和break,那么这2者的区别是?continue是跳出本次循环,进行下一次循环;break是跳出整个循环;例如:...

    简单学Python——关键字6——break和continue

    Python退出循环,有break语句和continue语句两种实现方式。break语句和continue语句的区别:break语句作用是终止循环。continue语句作用是跳出本轮循环,继续下一次循...

    2-1,0基础学Python之 break退出循环、 continue继续循环 多重循

    用for循环或者while循环时,如果要在循环体内直接退出循环,可以使用break语句。比如计算1至100的整数和,我们用while来实现:sum=0x=1whileTrue...

    Python 中 break 和 continue 傻傻分不清

    大家好啊,我是大田。今天分享一下break和continue在代码中的执行效果是什么,进一步区分出二者的区别。一、continue例1:当小明3岁时不打印年龄,其余年龄正常循环打印。可以看...

    python中的流程控制语句:continue、break 和 return使用方法

    Python中,continue、break和return是控制流程的关键语句,用于在循环或函数中提前退出或跳过某些操作。它们的用途和区别如下:1.continue(跳过当前循环的剩余部分,进...

    L017:continue和break - 教程文案

    continue和break在Python中,continue和break是用于控制循环(如for和while)执行流程的关键字,它们的作用如下:1.continue:跳过当前迭代,...

    作为前端开发者,你都经历过怎样的面试?

    已经裸辞1个月了,最近开始投简历找工作,遇到各种各样的面试,今天分享一下。其实在职的时候也做过面试官,面试官时,感觉自己问的问题很难区分候选人的能力,最好的办法就是看看候选人的github上的代码仓库...

    面试被问 const 是否不可变?这样回答才显功底

    作为前端开发者,我在学习ES6特性时,总被const的"善变"搞得一头雾水——为什么用const声明的数组还能push元素?为什么基本类型赋值就会报错?直到翻遍MDN文档、对着内存图反...

    2023金九银十必看前端面试题!2w字精品!

    导文2023金九银十必看前端面试题!金九银十黄金期来了想要跳槽的小伙伴快来看啊CSS1.请解释CSS的盒模型是什么,并描述其组成部分。答案:CSS的盒模型是用于布局和定位元素的概念。它由内容区域...

    前端面试总结_前端面试题整理

    记得当时大二的时候,看到实验室的学长学姐忙于各种春招,有些收获了大厂offer,有些还在苦苦面试,其实那时候的心里还蛮忐忑的,不知道自己大三的时候会是什么样的一个水平,所以从19年的寒假放完,大二下学...

    由浅入深,66条JavaScript面试知识点(七)

    作者:JakeZhang转发链接:https://juejin.im/post/5ef8377f6fb9a07e693a6061目录由浅入深,66条JavaScript面试知识点(一)由浅入深,66...

    2024前端面试真题之—VUE篇_前端面试题vue2020及答案

    添加图片注释,不超过140字(可选)1.vue的生命周期有哪些及每个生命周期做了什么?beforeCreate是newVue()之后触发的第一个钩子,在当前阶段data、methods、com...

    今年最常见的前端面试题,你会做几道?

    在面试或招聘前端开发人员时,期望、现实和需求之间总是存在着巨大差距。面试其实是一个交流想法的地方,挑战人们的思考方式,并客观地分析给定的问题。可以通过面试了解人们如何做出决策,了解一个人对技术和解决问...