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

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

zhezhongyun 2025-04-11 09:45 28 浏览

在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

    相关推荐

    激光手术矫正视力对眼睛到底有没有伤害?

    因为大家询问到很多关于“基质不能完全愈合”的问题,有必要在这里再详细解释一下。谢谢@珍惜年少时光提出的疑问:因为手头刚好在看组织学,其中提到:”角膜基质约占角膜的全厚度的90%,主要成分是胶原板层,...

    OneCode核心概念解析——View(视图)

    什么是视图?在前面的章节中介绍过,Page相关的概念,Page是用户交互的入口,具有Url唯一性。但Page还只是一个抽象的容器,而View则是一个具备了具体业务能力的特殊的Page,它可以是一个...

    精品博文图文详解Xilinx ISE14.7 安装教程

    在软件安装之前,得准备好软件安装包,可从Xilinx官网上下载:http://china.xilinx.com/support/download/index.html/content/xilinx/z...

    卡片项目管理(Web)(卡片设计的流程)

    简洁的HTML文档卡片管理,简单框架个人本地离线使用。将个人工具类的文档整理使用。优化方向:添加图片、瀑布式布局、颜色修改、毛玻璃效果等。<!DOCTYPEhtml><html...

    GolangWeb框架Iris项目实战-JWT和中间件(Middleware)的使用EP07

    前文再续,上一回我们完成了用户的登录逻辑,将之前用户管理模块中添加的用户账号进行账号和密码的校验,过程中使用图形验证码强制进行人机交互,防止账号的密码被暴力破解。本回我们需要为登录成功的用户生成Tok...

    sitemap 网站地图是什么格式?有什么好处?

    sitemap网站地图方便搜索引擎发现和爬取网页站点地图是一种xml文件,或者是txt,是将网站的所有网址列在这个文件中,为了方便搜索引擎发现并收录的。sitemap网站地图分两种:用于用户导...

    如何在HarmonyOS NEXT中处理页面间的数据传递?

    大家好,前两天的Mate70的发布,让人热血沸腾啊,不想错过,自学的小伙伴一起啊,今天分享的学习笔记是关于页面间数据伟递的问题,在HarmonyOSNEXT5.0中,页面间的数据传递可以有很多种...

    从 Element UI 源码的构建流程来看前端 UI 库设计

    作者:前端森林转发链接:https://mp.weixin.qq.com/s/ziDMLDJcvx07aM6xoEyWHQ引言由于业务需要,近期团队要搞一套自己的UI组件库,框架方面还是Vue。而业界...

    jq+ajax+bootstrap改了一个动态分页的表格

    最近在维护一个很古老的项目,里面是用jq的dataTable方法实现一个分页的表格,不过这些表格的分页是本地分页。现在想要的是点击分页去请求数据。经过多次的修改,以失败告终。分页的不准确,还会有这个错...

    学习ES6- 入门Vue(大量源代码及笔记,带你起飞)

    ES6学习网站:https://es6.ruanyifeng.com/箭头函数普通函数//普通函数this指向调用时所在的对象(可变)letfn=functionfn(a,b){...

    青锋微服务架构之-Ant Design Pro 基本配置

    青锋(msxy)-Gitee.com1、更换AntDesignPro的logo和名称需要修改文件所在位置:/config/defaultSetting.jsconstproSett...

    大数据调度服务监控平台(大数据调度服务监控平台官网)

    简介SmartKettle是针对上述企业的痛点,对kettle的使用做了一些包装、优化,使其在web端也能具备基础的kettle作业、转换的配置、调度、监控,能在很大一定程度上协助企业完成不同...

    Flask博客实战 - 实现博客首页视图及样式

    本套教程是一个Flask实战类教程,html/css/javascript等相关技术栈不会过多的去详细解释,那么就需要各位初学者尽可能的先去掌握这些基础知识,当然本套教程不需要你对其非常精通,但最起码...

    Web自动化测试:模拟鼠标操作(ActionChains)

    在日常的测试中,经常会遇到需要鼠标去操作的一些事情,比如说悬浮菜单、拖动验证码等,这一节我们来学习如何使用webdriver模拟鼠标的操作首页模拟鼠标的操作要首先引入ActionChains的包fro...

    DCS F-16C 中文指南 16.9ILS仪表降落系统教程

    10–ILS教程我们的ILS(仪表着陆进近)将到达Batumi巴统机场。ILS频率:110.30跑道航向:120磁航向/126真航向无线电塔频率:131.0001.设置雷达高度表开关打开(前)并...