当前位置: 首页 > news >正文

当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,因为浅拷贝共享了嵌套对象!

http://www.cnnetsun.cn/news/131829.html

相关文章:

  • 北京创业省钱攻略!0 元注册公司不是梦
  • 行业标杆 | 越秀地产入选「2025年度数据湖仓应用创新先锋企业」
  • 云边 Agent 延迟优化全攻略(99%工程师忽略的底层机制曝光)
  • 【物流运输Agent路线优化】:揭秘智能路径调整背后的算法黑科技
  • 终于搞懂了!React Agent的实现思路原来这么简单,核心就这3步!
  • FLORIS风电场仿真终极指南:10个核心技巧快速掌握工程尾流模型
  • 工业HMI常见故障及排查方法:黑屏、通信中断、操作无响应,新手别慌
  • 55页PDF信息系统项目管理师考神笔记
  • 工业HMI的发展趋势:智能化、一体化、移动化,新手该怎么跟上?
  • 【高频卫星信号处理突破】:3步实现Agent端低延迟响应
  • 联想拯救者BIOS隐藏功能一键解锁:3分钟搞定高级设置终极教程
  • 从功能测试到自动化转型:测试工程师的进阶之路
  • 测试与开发的协同工作模式:从对立到共生的范式演进
  • VCAM安卓虚拟相机:打造你的专属视频魔法师
  • 抢占本地生活服务市场 开源可定制的跑腿平台源码推荐
  • 【物流量子 Agent 成本优化】:揭秘降低运营开支的5大核心技术策略
  • 企业微信Webhook Java SDK:从零构建智能消息推送系统
  • 跨境独立站+海淘代购运营指南
  • OPC-UA客户端终极实战指南:从零掌握工业数据交互
  • 从对话到共情,元宇宙Agent如何实现自然交互?
  • Windows与Linux环境兼容难题,MCP PL-600 Agent如何无缝部署?
  • (AZ-500云Agent深度研究):从注册流程到策略执行的完整技术链剖析
  • CanOpenSTM32深度解析:STM32微控制器的CANopen协议栈架构揭秘
  • 深入NVIDIA Nemotron-3:高效准确的技术、工具与数据深度解析
  • 如何用LD2410雷达传感器实现3种智能人体检测方案
  • HTML极致压缩完整指南:如何快速提升网站加载速度
  • 量子计算如何彻底改变量子力学?
  • 管理制造企业的5种方法
  • 面向自然科学的人工智能建模方法
  • 基于VUE的南山社区图书管理系统[VUE]-计算机毕业设计源码+LW文档