经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » JS/JS库/框架 » JavaScript » 查看文章
js中如何复制一个数组(浅复制、深复制)
来源:jb51  时间:2023/4/14 9:17:53  对本文有异议

在js中,我们经常会用到数组复制,Array是引用类型,如果用arrA=arrB简单的把一个数组赋值,并没有创造出一个新数组,arrA和arrB其实指向的还是同一个地址,改变一个另一个也会随之改变,很明显这并不是我们想要的

  1. ? var arr = [1, 2, 3];
  2. ? var newArr = arr;
  3. ? arr.push(4);
  4. ? console.log(newArr1); ?// [1, 2, 3, 4]

下面介绍数组的浅复制

(适用于数组并不复杂,即数组中没有嵌套对象或者嵌套数组)

方法一:concat()

  • concat()方法用于连接两个或多个数组;
  • concat() 方法不会更改现有数组,而是返回一个新数组,其中包含已连接数组的值。
  1. ? var arr = [1, 2, 3];
  2. ? var newArr = arr.concat();
  3. ? arr.push(4);
  4. ? console.log(newArr); // [1, 2, 3]

方法二:slice()

  • slice() 方法以新的数组对象,返回数组中被选中的元素;
  • slice() 方法选择从给定的 start 参数开始的元素,并在给定的 end 参数处结束,但不包括;
  • slice() 方法不会改变原始数组;
  1. ? var arr = [1, 2, 3];
  2. ? var newArr = arr.slice();
  3. ? arr[0] = 10;
  4. ? console.log(arr);// [10, 2, 3]
  5. ? console.log(newArr);// [1, 2, 3]

方法三:扩展运算符

  1. ? var arr = [1, 2, 3];
  2. ? var [ ...newArr ] = arr;
  3. ? arr[0] = 10;
  4. ? console.log(arr); // [10, 2, 3]
  5. ? console.log(newArr);// [1, 2, 3]

方法四: Object.assign()

  1. ?var arr = [1, 2, 3];
  2. ?var newArr = Object.assign([], arr);
  3. ?arr[0] = 10;
  4. ?console.log(arr);// [10, 2, 3]
  5. ?console.log(newArr);// [1, 2, 3]

如果数组元素是对象或者数组,上面四种方法就会只拷贝数组或者对象的引用,如果我们对其中一个数组进行修改,另一个数组也会发生变化

比如:

  1. ? var arr = [ { a: 1 }, [ 1, 2 ], 3 ];
  2. ? let newArr = arr.concat();
  3. ? arr[0].a = 2;
  4. ? console.log(arr); // [ { a: 2 }, [ 1, 2 ], 3 ]
  5. ? console.log(newArr);// [ { a: 2 }, [ 1, 2 ], 3 ] 值被影响

下面是深复制

(可以完全拷贝一个数组,即使嵌套了对象或者数组,两者也不会互相影响)

方法一:JSON.parse(JSON.stringify(arr))

  1. ? var arr = [ { a: 1 }, [ 1, 2 ], 3 ];
  2. ? // let newArr = JSON.parse(JSON.stringify(arr));
  3. ? let newArr = arr.concat();
  4. ? arr[0].a = 2;
  5. ? console.log(arr); // [ { a: 2 }, [ 1, 2 ], 3 ]
  6. ? console.log(newArr);// [ { a: 1 }, [ 1, 2 ], 3 ]

但是该方法是有局限性的

  • 会忽略 undefined
  • 会忽略 symbol
  • 不能序列化函数
  • 不能解决循环引用的对象

比如下面这个例子:

  1. let a = {
  2. ? age: undefined,
  3. ? sex: Symbol('male'),
  4. ? jobs: function() {},
  5. ? name: 'sun'
  6. }
  7. let b = JSON.parse(JSON.stringify(a))
  8. console.log(b) // {name: "sun"}

方法二:通用方法(数组或对象)

拷贝的时候判断属性值的类型,如果是对象,继续递归调用深拷贝函数(简易版)

  1. ? var deepCopy = function(obj) {
  2. ? ? // 判断是否是对象
  3. ? ? if (typeof obj !== 'object') return;
  4. ? ? // 判断obj类型,根据类型新建一个对象或者数组
  5. ? ? var newObj = obj instanceof Array ? [] : {}
  6. ? ? // 遍历对象,进行赋值
  7. ? ? for (var key in obj) {
  8. ? ? ? if (obj.hasOwnProperty(key)) {
  9. ? ? ? ? let val = obj[key];
  10. ? ? ? ? // 判断属性值的类型,如果是对象,递归调用deepCopy
  11. ? ? ? ? newObj[key] = typeof val === 'object' ? deepCopy(val) : val
  12. ? ? ? }
  13. ? ? }
  14. ? ? return newObj
  15. ? }

方法三:利用lodash的深拷贝函数

  1. _.cloneDeep(value)

其中value就是要深拷贝的值

简单例子

  1. var objects = [{ 'a': 1 }, { 'b': 2 }];
  2. var deep = _.cloneDeep(objects);
  3. console.log(deep[0] === objects[0]);
  4. // => false

在Vue中使用

安装

  1. npm i --save lodash

在main.js中引入

  1. import _ from 'lodash';
  2. Vue.prototype._ = _;

使用

  1. let newObj = this._.cloneDeep(oldObj)

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持w3xue。

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号