news 2026/6/23 16:59:45

当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/6/23 15:22:38

北京创业省钱攻略!0 元注册公司不是梦

北京创业省钱攻略!0 元注册公司不是梦 谁说在北京开公司要花大几千?亲身实测,核名、执照办理、刻章备案全程 0 费用!工作人员一句话直接让我震惊:“现在北京对初创者的支持力度就是这么大”,当场截图发圈分…

作者头像 李华
网站建设 2026/6/23 3:24:31

行业标杆 | 越秀地产入选「2025年度数据湖仓应用创新先锋企业」

近日,镜舟科技发布“2025 年度数据湖仓应用创新先锋企业”奖项,作为中国第一代商品房缔造者和全国综合性房企领军者,越秀地产凭借其在数据湖仓建设与应用方面的卓越实践成功入选,为房地产行业的数字化转型树立了标杆。一、越秀地产…

作者头像 李华
网站建设 2026/6/22 16:40:40

云边 Agent 延迟优化全攻略(99%工程师忽略的底层机制曝光)

第一章:云边 Agent 延迟优化的核心挑战在云计算与边缘计算协同演进的背景下,云边 Agent 作为连接云端控制平面与边缘设备的关键组件,其响应延迟直接影响整体系统的实时性与可靠性。由于边缘节点分布广泛、网络环境复杂,Agent 在任…

作者头像 李华
网站建设 2026/6/23 10:50:47

【物流运输Agent路线优化】:揭秘智能路径调整背后的算法黑科技

第一章:物流运输Agent路线调整的核心挑战在动态复杂的物流网络中,运输Agent的路线调整面临多重技术与现实约束的交织挑战。系统需实时响应交通状况、天气变化、订单变更等外部扰动,同时保证整体配送效率与成本控制之间的平衡。环境不确定性带…

作者头像 李华
网站建设 2026/6/23 4:19:30

终于搞懂了!React Agent的实现思路原来这么简单,核心就这3步!

最近,国内AI领域创新速度不断刷新记录,卷出了新高度。近两周诞生了n款颠覆性的开源大模型,在智能体(Agent)和深度研究(DeepResearch)方向也同样,几乎每隔一段时间就有新产品或新功能…

作者头像 李华
网站建设 2026/6/21 8:01:25

FLORIS风电场仿真终极指南:10个核心技巧快速掌握工程尾流模型

FLORIS风电场仿真终极指南:10个核心技巧快速掌握工程尾流模型 【免费下载链接】floris A controls-oriented engineering wake model. 项目地址: https://gitcode.com/gh_mirrors/fl/floris 在风电场开发与运营过程中,如何准确预测多风机间的尾流…

作者头像 李华