news 2026/6/15 8:21:56

当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/15 7:31:43

谷歌放大招!Gemini 3 Flash:地表最强“性价比之王“诞生

开篇:AI界的"价格屠夫"来了 如果有人告诉你,现在有一个AI模型,它比GPT-5.2便宜3倍,比Claude Sonnet 4.5便宜6倍,速度还快得飞起,编码能力甚至超过了自家的Pro版本——你会不会觉得这是在吹牛&am…

作者头像 李华
网站建设 2026/6/14 19:27:49

【企业级通信安全升级】:MCP MS-720 Agent更新为何刻不容缓

第一章:MCP MS-720 Agent更新的紧迫性随着企业IT基础设施的持续演进,终端安全与系统兼容性问题日益突出。MCP MS-720 Agent作为关键的设备管理组件,其版本滞后可能导致认证失败、策略同步异常以及远程控制功能失效。近期多起生产环境故障案例…

作者头像 李华
网站建设 2026/6/14 20:50:57

MCP SC-400补丁管理陷阱,99%运维人员都踩过的坑该如何避免?

第一章:MCP SC-400补丁管理的核心挑战在现代企业IT环境中,MCP SC-400补丁管理面临多重复杂性,尤其是在混合云架构和异构操作系统共存的场景下。补丁的及时部署不仅关乎系统稳定性,更直接影响安全合规性与数据保护能力。补丁分发的…

作者头像 李华
网站建设 2026/6/12 16:14:09

强力突破百度网盘限速:Mac端开源插件完整解决方案

还在为百度网盘那蜗牛般的下载速度而苦恼吗?😫 作为Mac用户,你一定经历过重要文件下载到一半却因为限速而陷入漫长等待的窘境。传统的优化方法要么操作复杂,要么存在安全风险,而官方提供的极速下载试用更是转瞬即逝&am…

作者头像 李华
网站建设 2026/6/12 4:44:06

电力系统负荷预测新突破:Agent智能模型如何提升预测精度90%?

第一章:电力系统负荷预测新突破:Agent智能模型如何提升预测精度90%?在现代智能电网的运行中,电力负荷预测的准确性直接影响调度效率与能源成本。传统统计模型如ARIMA或SVM在面对非线性、高波动性的用电数据时表现受限。近年来&…

作者头像 李华
网站建设 2026/6/13 19:11:09

【元宇宙Agent交互升级指南】:掌握3种主流通信协议与状态同步模型

第一章:元宇宙 Agent 的交互逻辑 在元宇宙环境中,Agent(智能体)作为用户代理或自主实体,其核心功能依赖于复杂的交互逻辑。这些逻辑决定了 Agent 如何感知环境、处理信息并作出响应。一个典型的交互流程包含感知、决策…

作者头像 李华