数组的深拷贝问题方法详解

分析

深拷贝就是将数据在堆中进行了拷贝的结果,这样对复制过后的对象的操作便不会影响到原对象。

有如下方法
使用解构赋值 
使用 Object.create
使用 JSON.parse 和 JSON.stringify
使用 structuredClone
使用第三方库,例如 lodash
自己书写方法实现

方法

1、解构赋值有个问题,只能复制对象内第一层的值,更深一层便无能为力了。如([...arr])
2、使用 Object.create (let personobj = Object.create(person)) Object.create 对于数组的深拷贝不理想
Object.create 使用现有的对象来创建一个新的对象。同样的,也只能解决第一层的复制。
3、JSON.parse 和 JSON.stringify:先用 JSON.stringify 将对象序列化成字符串,再用 JSON.parse 解析回来,可以达到深拷贝的效果,并且对于多层对象同样有效。
但是对于对象中有用户定义的类型无效(如:obj={
    path: 'login',
    name: 'login',
    meta: {
      i18n: 'menu.login',
      name: i18n.t('menu.login'),
      disabled: true,
    },
    component: () => import('@/views/Login.vue'),
  })里面的i18n.t语言切换,component属性都不能有效拷贝
  4、全局的 structuredClone() 方法使用结构化克隆算法将给定的值进行深拷贝。
  但同样对方法或自定义对象无力。使用方法(https://developer.mozilla.org/zh-CN/docs/web/api/structuredClone)
  5、使用第三方库,例如 lodash 官网(https://www.lodashjs.com/docs/lodash.cloneDeep#_clonedeepvalue)
如(_.cloneDeep(person))
6、自身书写实现如下

深拷贝方法封装

const copyList = (arr: any) => {
  const result = []
  for (const item of arr) {
    result.push(deepclone(item))
  }
  return result
}
const deepclone: any = (obj: any) => {
  if (typeof obj === 'object') {
    // console.log(1, '是否死循环')
    if (Array.isArray(obj)) {
      return copyList(obj)
    } else {
      const result: any = {}
      for (const key in obj) {
        result[key] = deepclone(obj[key])
      }
      return result
    }
  } else {
    return obj
  }
}
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
赋值是指将一个变量的值赋给另一个变量,二者指向同一个内存地址。示例代码如下: ``` let a = [1, 2, 3]; let b = a; console.log(a); // [1, 2, 3] console.log(b); // [1, 2, 3] b.push(4); console.log(a); // [1, 2, 3, 4] console.log(b); // [1, 2, 3, 4] ``` 上述代码中,由于b直接指向了a的内存地址,所以对b进行修改也会影响到a。这就是赋值的情况。 浅拷贝是指创建一个新的对象或数组,但只复制了对象或数组的引用,而不是复制其中的内容。示例代码如下: ``` let obj = {name: 'Alice', age: 20}; let copy = Object.assign({}, obj); console.log(obj); // {name: 'Alice', age: 20} console.log(copy); // {name: 'Alice', age: 20} copy.age = 25; console.log(obj); // {name: 'Alice', age: 20} console.log(copy); // {name: 'Alice', age: 25} ``` 在上述代码中,通过`Object.assign`方法将obj的属性复制到新对象copy中,但是只是复制了引用,并没有复制其中的属性值。因此,对copy进行修改不会影响到原来的obj对象。 深拷贝是指创建一个新的对象或数组,并且递归地复制了对象或数组的所有内容,包括嵌套的对象和数组。示例代码如下: ``` let arr = [1, 2, [3, 4]]; let deepCopy = JSON.parse(JSON.stringify(arr)); console.log(arr); // [1, 2, [3, 4]] console.log(deepCopy); // [1, 2, [3, 4]] deepCopy[2][0] = 5; console.log(arr); // [1, 2, [3, 4]] console.log(deepCopy); // [1, 2, [5, 4]] ``` 在上述代码中,通过`JSON.stringify`和`JSON.parse`方法实现了深拷贝。新的deepCopy数组完全独立于原来的arr数组,因此对deepCopy进行修改不会影响到原来的arr数组。 总结起来,赋值只是简单地将变量的值赋给另一个变量,二者指向同一个内存地址;浅拷贝只复制了对象或数组的引用,对新对象的修改会影响到原始对象;深拷贝递归地复制了对象或数组的所有内容,新对象与原始对象完全独立。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值