news 2026/4/27 8:19:19

当JS拷贝玩起了“俄罗斯套娃”:深拷贝与浅拷贝的趣味对决

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
当JS拷贝玩起了“俄罗斯套娃”:深拷贝与浅拷贝的趣味对决

欢迎使用我的小程序👇👇👇👇 俱好用助手功能介绍


📚 拷贝不只是复制粘贴

想象一下,你有一本心爱的精装书,朋友想借去阅读。你有两个选择:

  1. 直接给朋友- 但书就不在你手上了(原始引用)
  2. 去复印店复印一本- 朋友有自己的副本,你的原版还在

在JavaScript的世界里,拷贝数据就像这个场景,但有更多的“套娃”情况!

🎭 浅拷贝:只搬家的“表面朋友”

浅拷贝就像只搬走了家具,但墙上还挂着原房子的钥匙:

// 原始对象 - 一个“套娃”对象constoriginal={name:"小明",age:25,hobbies:["编程","游戏","阅读"],// 注意这个数组!address:{city:"北京",district:"海淀区"}};// 浅拷贝的几种方式:constshallowCopy1=Object.assign({},original);constshallowCopy2={...original};// 扩展运算符constshallowCopy3=original.slice?.();// 数组专用// 试试修改会发生什么?shallowCopy1.name="小红";// ✅ 不会影响originalshallowCopy1.hobbies.push("摄影");// ⚠️ 糟糕!original的hobbies也被改了!

浅拷贝的特点:

  • 只复制第一层属性
  • 嵌套对象/数组仍然是“共享”的引用
  • 像只换了外壳,芯子还是同一个

🧳 深拷贝:真正的“独立门户”

深拷贝就像带着所有家当搬到了全新的房子:

// 深拷贝实现方式大比拼// 方法1:JSON大法(最简单但有局限)constdeepCopy1=JSON.parse(JSON.stringify(original));// 方法2:递归实现(自己动手,丰衣足食)functiondeepClone(obj){if(obj===null||typeofobj!=='object')returnobj;if(objinstanceofDate)returnnewDate(obj);if(objinstanceofRegExp)returnnewRegExp(obj);constclone=Array.isArray(obj)?[]:{};for(letkeyinobj){if(obj.hasOwnProperty(key)){clone[key]=deepClone(obj[key]);}}returnclone;}constdeepCopy2=deepClone(original);// 方法3:使用现成库(最省心)// const deepCopy3 = _.cloneDeep(original); // Lodash// const deepCopy4 = structuredClone(original); // 现代JS原生API

现在无论怎么修改deepCopy,都不会影响original了!

🎯 什么时候用什么?

浅拷贝适用场景:

// 场景1:配置合并constdefaultConfig={theme:'light',showTips:true};constuserConfig={theme:'dark'};constfinalConfig={...defaultConfig,...userConfig};// 场景2:创建对象副本进行简单修改constuser={name:'张三',loggedIn:false};constupdatedUser={...user,loggedIn:true};

深拷贝适用场景:

// 场景1:状态管理(如Redux reducer)functionreducer(state,action){switch(action.type){case'UPDATE_USER':return{...state,user:deepClone(action.payload)// 确保完全独立};}}// 场景2:表单数据的初始副本constinitialFormData=deepClone(templateData);constformData=deepClone(initialFormData);// 每次都是新的开始

🧪 趣味实验:拷贝的陷阱

// 陷阱1:循环引用(自己引用自己)constnarcissist={name:"自恋对象"};narcissist.self=narcissist;// 我引用我自己!// JSON大法会报错!// JSON.parse(JSON.stringify(narcissist)); // TypeError!// 陷阱2:特殊对象constspecialObj={date:newDate(),regex:/hello/gi,func:function(){return"Hi";},undefined:undefined,infinity:Infinity,nan:NaN};console.log(JSON.parse(JSON.stringify(specialObj)));// 函数、undefined不见了!日期变成了字符串...

📊 性能对比:速度与深度的博弈

方法速度深度特殊类型支持循环引用支持
扩展运算符...⚡⚡⚡⚡⚡有限
Object.assign()⚡⚡⚡⚡⚡有限
JSON方法⚡⚡⚡
递归实现⚡⚡
Lodash的cloneDeep⚡⚡很好
structuredClone()⚡⚡⚡较好

💡 实用小贴士

  1. “先问要不要,再问怎么做”- 先确定是否需要深拷贝,很多情况浅拷贝就够用了

  2. 现代JS的救星

    // 浏览器和Node.js的新宠constcloned=structuredClone(original);// 支持大部分类型!
  3. Lodash是你的好朋友

    import{cloneDeep}from'lodash-es';// 按需引入// 或者用 throttle 的深拷贝函数
  4. 性能提示:对于超大对象,考虑是否需要整个拷贝,也许只需修改部分

🎬 实战演练:拷贝在真实场景的应用

// 购物车场景constshoppingCart={items:[{id:1,name:"JavaScript高级编程",quantity:1,price:99},{id:2,name:"TypeScript入门",quantity:2,price:79}],discount:0.1,getTotal(){returnthis.items.reduce((sum,item)=>sum+item.price*item.quantity,0)*(1-this.discount);}};// 用户想“如果这样买”的试算功能functionwhatIfAddItem(cart,newItem){consthypotheticalCart=deepClone(cart);hypotheticalCart.items.push(newItem);returnhypotheticalCart.getTotal();}// 真实的购物车不受影响console.log(shoppingCart.items.length);// 还是2个

📝 总结:拷贝选择指南

  1. “我只想改改表面”→ 用浅拷贝(...Object.assign
  2. “我要完全独立的新对象”→ 用深拷贝
  3. “我有特殊类型或循环引用”→ 用structuredClone()或Lodash
  4. “不确定深浅”→ 问问自己:嵌套对象需要独立吗?

记住,在JavaScript的世界里,“拷贝”不是简单的复制粘贴,而是关于“独立性”的哲学选择。选择正确的拷贝方式,能让你的代码更健壮、更可预测!

下次当你面对需要拷贝的场景时,不妨先想想:这是一个需要独立门户的深拷贝,还是一个可以共享芯子的浅拷贝?


小测验:你能看出下面代码的输出吗?

constobj={a:1,b:{c:2}};constshallow={...obj};constdeep=JSON.parse(JSON.stringify(obj));shallow.b.c=999;deep.b.c=888;console.log(obj.b.c);// 是多少?

答案:999,因为浅拷贝共享了嵌套对象!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/22 21:09:39

如何查询postgres数据库下的schema

方法一:使用SQL查询在连接到数据库后,可以执行以下SQL语句来查看所有schema:SELECT schema_name FROM information_schema.schemata;另外,PostgreSQL还提供了一个名为“pg_namespace”的系统表,也可以用来查询&#xf…

作者头像 李华
网站建设 2026/4/18 1:43:21

前端新人必看:CSS常用布局全解析(7天搞定+实战技巧)

前端新人必看:CSS常用布局全解析(7天搞定实战技巧)前端新人必看:CSS常用布局全解析(7天搞定实战技巧)为什么 CSS 布局是前端的“腰马合一”从 table 到 Grid:一部“翻车”史浮动布局&#xff1a…

作者头像 李华
网站建设 2026/4/26 0:41:31

GraphRAG从入门到精通:基于PolarDB+通义千问+LangChain,知识图谱与大模型融合实战,一篇就够了!

本文介绍了如何使用PolarDB、通义千问和LangChain搭建GraphRAG系统,结合知识图谱和向量检索提升问答质量。通过实例展示了单独使用向量检索和图检索的局限性,并通过图向量联合搜索增强了问答准确性。PolarDB支持AGE图引擎和pgvector插件,实现…

作者头像 李华
网站建设 2026/4/27 3:51:21

教育AI Agent的学习推荐引擎(从算法到落地的完整路径)

第一章:教育AI Agent的学习推荐引擎概述在现代在线教育系统中,个性化学习路径的实现依赖于智能推荐引擎的支持。教育AI Agent通过分析学习者的行为数据、知识掌握程度和兴趣偏好,动态生成定制化的学习内容推荐,从而提升学习效率与…

作者头像 李华
网站建设 2026/4/23 14:31:34

Python Wechaty微信机器人开发实战教程

Python Wechaty微信机器人开发实战教程 【免费下载链接】python-wechaty-getting-started Python Wechaty Starter Project Template that Works Out-of-the-Box 项目地址: https://gitcode.com/gh_mirrors/py/python-wechaty-getting-started 想要快速构建智能微信助手…

作者头像 李华
网站建设 2026/4/23 0:49:02

研发团队效率神器:手把手教你用蜘蛛表格构建自动化任务管理中枢

在研发团队的日常工作中,任务分配混乱、进度不透明、跨角色协作低效等问题常常成为项目推进的“绊脚石”。传统的Excel统计繁琐易错,专用项目管理工具又往往门槛高、配置复杂。本文将介绍一种轻量高效的解决方案——用蜘蛛表格快速搭建自动化研发任务看板…

作者头像 李华