JS【深浅 copy】

本贴最后更新于 1330 天前,其中的信息可能已经时移世改

computed: {1.浅拷贝是拷贝一层,深层次的对象级别的就只拷贝引用;

2.深拷贝是拷贝多层,每一级别的数据都会拷贝出来;

1.浅拷贝

computed: {
updateList () {
let newList = this.List
/*do something*/
return newList
},
}

本想的是设置个中间变量 newList,将它和原来的 List 相比做一些操作,最后返回这个 newList,后来发现原 List 也改变了。才意识到这个 newList 只是个引用类型,改变了它原来的对象也会变。

  查资料的过程中,注意到了还有深拷贝浅拷贝的区别:

2.深拷贝

slice()方法

对于array对象的slice函数,返回一个数组的一段。(仍为数组)
arrayObj.slice(start, [end])

参数:
arrayObj 必选项。一个 Array 对象。
start 必选项。arrayObj 中所指定的部分的开始元素是从零开始计算的下标。
end可选项。arrayObj 中所指定的部分的结束元素是从零开始计算的下标。

说明:
slice 方法返回一个 Array 对象,其中包含了 arrayObj 的指定部分。
slice 方法一直复制到 end 所指定的元素,但是不包括该元素。
如果 start 为负,将它作为 length + start处理,此处 length 为数组的长度。
如果 end 为负,就将它作为 length + end 处理,此处 length 为数组的长度。
如果省略 end ,那么 slice 方法将一直复制到 arrayObj 的结尾。
如果 end 出现在 start 之前,不复制任何元素到新数组中。

对于本例,let newList = this.List.slice()即可

concat()方法

concat() 方法用于连接两个或多个数组。该方法不会改变现有的数组,而仅仅会返回被连接数组的一个副本。
语法:arrayObject.concat(arrayX,arrayX,......,arrayX)
说明:返回一个新的数组。该数组是通过把所有 arrayX 参数添加到 arrayObject 中生成的。如果要进行 concat() 操作的参数是数组,那么添加的是数组中的元素,而不是数组。

对于本例,let newList = this.List.concat()即可

但是,这两种方法都有局限性,如:

var arr1 = [{"name":"weifeng"},{"name":"boy"}];//原数组
var arr2 = [].concat(arr1);//拷贝数组
arr1[1].name="girl";
console.log(arr1);// [{"name":"weifeng"},{"name":"girl"}]
console.log(arr2);//[{"name":"weifeng"},{"name":"girl"}]



var a1=[["1","2","3"],"2","3"],a2;
a2=a1.slice(0);
a1[0][0]=0; //改变a1第一个元素中的第一个元素
console.log(a2[0][0]);  //影响到了a2

var b1=[["1","2","3"],"2","3"],b2;
b2=b1.slice(0);
b1[0][0]=0; //改变a1第一个元素中的第一个元素
console.log(b2[0][0]);  //影响到了a2

JSON.stringify 和 JSON.parse 实现深拷贝:

JSON.stringify 把对象转成字符串,再用 JSON.parse 把字符串转成新的对象

对于本例, let newList = JSON.parse(JSON.stringify(this.List))

JSON.stringify()有一些局限,比如不能拷贝 function:

JSON.stringify() - JavaScript | MDN This deprecated API should no longer be used, but will probably still work. This deprecated API should no longer be used, but will probably still work. This deprecated API should no longer be used, but will probably still work. This deprecated API should no longer be used, but will probably still work. This deprecated API should no longer be used, but will probably still work. This API has not been standardized. This is an obsolete API and is no longer guaranteed to work.

当然,如果是不那么复杂的数组,你可以声明一个新数组,自己写一个 for 循环拷贝过去。

  • JavaScript

    JavaScript 一种动态类型、弱类型、基于原型的直译式脚本语言,内置支持类型。它的解释器被称为 JavaScript 引擎,为浏览器的一部分,广泛用于客户端的脚本语言,最早是在 HTML 网页上使用,用来给 HTML 网页增加动态功能。

    729 引用 • 1327 回帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...