js 深拷贝、浅拷贝深度解析

赋值操作:

javascript 复制代码
    let obj={
      a:1,
      b:[1,2,3],
      c:{
        m:2
      }
    }
    let newObj=obj
    newObj.a=2
    newObj.b.push(4)
    newObj.c.m=3
    console.log(obj,newObj);

将一个对象赋值给一个变量,其实就是将这个对象在栈内存中的引用地址复制给了这个变量,这两个对象指向堆内存中的同一个数据引用,所以,其中一个对象的值发生变化,另一个都会跟着变化。

浅拷贝的实现方法:

  1. Object.assign()

  2. Array.prototype.concat()

  3. Array.prototype.slice()

  4. 对象和数组的扩展运算符:{...obj},...arr

javascript 复制代码
    let obj={
      a:1,
      b:[1,2,3],
      c:{
        m:2
      },
    }
    const newObj={...obj}
    obj.a=3
    obj.b.push(4)
    console.log(obj,newObj);

浅拷贝会在堆内存中开辟新的空间。浅拷贝只复制对象的第一层属性值,如果属性值是基本类型(如数字、字符串、布尔值),则直接复制其值;如果属性值是引用类型(如数组、对象),则复制其引用地址,而不是创建一个新的对象或数组。因此,浅拷贝后的对象和原对象共享相同的引用类型的属性值,修改其中一个对象的这些属性值会影响到另一个对象。

深拷贝实现方法:

  1. JSON.parse(JSON.stringify())
javascript 复制代码
    let obj={
      a:1,
      b:[1,2,3],
      c:{
        m:2
      }
    }
    let newObj=JSON.parse(JSON.stringify(obj))
    newObj.a=2
    newObj.b.push(4)
    newObj.c.m=3
    console.log(obj,newObj);// 输出结果 obj的数据没有变化
javascript 复制代码
    let obj={
      a:null,
      b:[1,2,3],
      c:undefined,
      fn:()=>{console.log(123);}
    }
    let newObj=JSON.parse(JSON.stringify(obj))
    newObj.b.push(4)
    console.log(obj,newObj);

JSON.parse(JSON.stringify()) 可以实现深拷贝,缺点是,如果对象或数组中有函数或undefined,就会丢失。

  1. 封装深拷贝函数
javascript 复制代码
    let obj={
      a:1,
      b:[1,2,3],
      c:{
        m:2
      },
      k:undefined,
      g:null,
      fn:()=>{console.log(123);}
    }
    function deepCloneFunc(obj) {
      if(typeof obj !=='object') return obj
      const newObj=obj instanceof Array?[]:{}
      for(let key in obj){
        if(obj.hasOwnProperty(key)){
          const value = obj[key]
          newObj[key]=typeof value ==='object'?deepCloneFunc(value):value
        }
      }
      return newObj
    }
    const newObj=deepCloneFunc(obj)
    console.log(newObj);

深拷贝其实就是在堆内存中重新开辟了一块内存,创建了一个新的对象,这个对象和被拷贝的对象用于相同的属性和值,他们两个在栈内存中的引用地址也不一样,是相互独立的,因此修改任何一个对象的值,另一个都不会跟着变化。

相关推荐
weedsfly1 分钟前
还在用 Axios?你可能需要重新理解 XHR 与 Fetch
前端·javascript·面试
CoderWeen3 分钟前
从零实现一个 Vue3 流程图编辑器:节点拖拽、贝塞尔连线与框选
前端·javascript
To_OC10 小时前
LC 128 最长连续序列:别上来就排序,O (n) 解法才是这题的灵魂
javascript·算法·leetcode
kyriewen15 小时前
我用 50 行代码重写了 React Router 核心,终于搞懂了前端路由原理
前端·javascript·react.js
Asize17 小时前
HTML5 Canvas 基础:从按帧动画到 ECharts 数据可视化
前端·javascript·canvas
默_笙17 小时前
🎄 后端给我一堆扁平数据,我 10 行代码把它变成了树
前端·javascript
前端Hardy17 小时前
又一个 AI 神器火了!
前端·javascript·后端
PBitW18 小时前
GPT训练我的第二天,我表示不过如此!!!😕😕😕
前端·javascript·面试
kyriewen19 小时前
白宫直接给 OpenAI 下了限制令,GPT-5.6 不能随便放出来了
前端·javascript·面试
默_笙1 天前
🍞 我用 CSS 画了一个会转的 3D 立方体,同事以为我学了 Three.js(这节课真的很神奇,我很喜欢)
javascript