js如何拷贝copy数组或对象

场景

在项目中经常有拷贝对象的情况,由于直接进行转移赋值只对基础类型的数据对象(eg:String,number)可以进行,对于引用类型的对象就不适用了,直接进行赋值只是对于指针的指向。

对于数组,且是简单数组(不是多层嵌套)的拷贝

对于简单的数组(不是多层嵌套)可以直接使用js的api就可以实现,相关api有:concat,slice,或者使用es6的…展开运算符。

对于对象,且是简单对象(不是多层嵌套)的拷贝

可以使用:Object.assign()方法或者使用es6的…展开运算符。

多层嵌套的复杂对象或数组深度拷贝

  • 使用JSON.parse和JSON.stringify实现拷贝
var obj1 = {a:{a:{a:1}},b:[1,2]};
var obj2 = JSON.parse(JSON.stringify(obj1));
console.log(obj2)
obj1.a.a=2;
obj2.a.a=3
console.log(obj1);
console.log(obj2);

这里可以实现深度拷贝,但由于JSON.stringify方法对于funtion、undefined、symbol会在转换中被忽略,所以会造成这些属性的缺失。

  • 使用深度遍历来进行实现拷贝
let _toString = Object.prototype.toString
//类型字典
let map = {
  array: 'Array',
  object: 'Object',
  function: 'Function',
  string: 'String',
  null: 'Null',
  undefined: 'Undefined',
  boolean: 'Boolean',
  number: 'Number'
}
//获取数据类型
let getType = (item) => {
  return _toString.call(item).slice(8, -1)
}
//判断类型
let isTypeOf = (item, type) => {
  return map[type] && map[type] === getType(item)
}

//visitedArr储存所有数据以判断是否存在环状数据
let deepClone = (obj, visitedArr = []) => {
  let _obj = {};
  if (isTypeOf(obj, 'array') || isTypeOf(obj, 'object')) {
    let index = visitedArr.indexOf(obj);
    _obj = isTypeOf(obj, 'array') ? [] : {};
    //~位非运算实际上就是对数字进行取负运算,再减 1
    if (~index) { // 判断环状数据
      _obj = visitedArr[index];
    } else {
      visitedArr.push(obj);
      for (let item in obj) {
      	//递归调用深度遍历
        _obj[item] = deepClone (obj[item], visitedArr);
      }
    }
  } else if (isTypeOf(obj, 'function')) {
    _obj = eval('(' + obj.toString() + ')');
  } else {
    _obj = obj;
  }
  return _obj;
}

总结

不同业务场景采用不同的方法,当然如果你使用了jquery可直接用$.extend方法来解决。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值