JSON

Javascript学习之浅拷贝和深拷贝详解

字号+ 作者:H5之家 来源:H5之家 2017-03-19 15:00 我要评论( )

Javascript学习之浅拷贝和深拷贝详解

标签:

JavaScript中,对于 Object  Array 这类引用类型值,当从一个变量向另一个变量复制引用类型值时,这个值的副本其实是一个指针,两个变量指向同一个堆对象,改变其中一个变量,另一个也会受到影响。

这种拷贝分为两种情况:拷贝引用和拷贝实例,也就是我们说的浅拷贝和深拷贝

浅拷贝(shallow copy

拷贝原对象的引用,这是最简单的浅拷贝。

// 对象var o1 = {a: 1};var o2 = o1;

console.log(o1 === o2);  // =>true

o2.a = 2; console.log(o1.a); // => 2

// 数组var o1 = [1,2,3];var o2 = o1;

console.log(o1 === o2); // => true

o2.push(4);console.log(o1); // => [1,2,3,4]

拷贝原对象的实例,但是对其内部的引用类型值,拷贝的是其引用,常用的就是如jquey中的$.extend({}, obj); Array.prototype.slice()  Array.prototype.concat() 都会返回一个数组或者对象的浅拷贝,举个例子:

var o1 = [’darko’, {age: 22}];var o2 = o1.slice(); // 根据Array.prototype.slice()的特性,这里会返回一个o1的浅拷贝对象

console.log(o1 === o2); // => false,说明o2拷贝的是o1的一个实例

o2[0] = ’lee’;console.log(o1[0]); // => "darko" o1o2内部包含的基本类型值,复制的是其实例,不会相互影响

o2[1].age = 23;console.log(o1[1].age); // =>23 o1o2内部包含的引用类型值,复制的是其引用,会相互影响

可以通过 Array.prototype.slice()  jQuery 中的 $.extend({}, obj) 完成对一个数组或者对象的浅拷贝,我们也可以自己写一个简单浅拷贝函数来加深对浅拷贝的理解、

// 浅拷贝实现,仅供参考function shallowClone(source) {

if (!source && typeof source !== ’object’) {

throw new Error(’error arguments’);

}

var targetObj = source.constructor === Array ? [] : {};

for (var keys in source) {

if (source.hasOwnProperty(keys)) {

targetObj[keys] = source[keys];

}

}

return targetObj;

}

深拷贝(deep copy

深拷贝也就是拷贝出一个新的实例,新的实例和之前的实例互不影响,深拷贝的实现有几种方法,首先我们可以借助jQuerylodash等第三方库完成一个深拷贝实例。在jQuery中可以通过添加一个参数来实现递归extend,调用 $.extend(true, {}, ...) 就可以实现一个深拷贝。

我们也可以自己实现一个深拷贝的函数,通常有两种方式,一种就是用 递归 的方式来做,还有一种是利用 JSON.stringify  JSON.parse 来做,这两种方式各有优劣,先来看看递归的方法怎么做。

jQuery中的extend方法基本的就是按照这个思路实现的,但是没有办法处理源对象内部 循环引用 的问题,同时对DateFuncion等类型值也没有实现真正的深度复制,但是这些类型的值在重新定义的时候一般都是直接覆盖,所以也不会对源对象产生影响,从一定程度上来说也算是实现了一个深拷贝。

// 递归实现一个深拷贝function deepClone(source){

if(!source && typeof source !== ’object’){

throw new Error(’error arguments’, ’shallowClone’);

}

var targetObj = source.constructor === Array ? [] : {};

for(var keys in source){

if(source.hasOwnProperty(keys)){

if(source[keys] && typeof source[keys] === ’object’){

targetObj[keys] = source[keys].constructor === Array ? [] : {};

targetObj[keys] = deepClone(source[keys]);

}else{

targetObj[keys] = source[keys];

}

}

}

return targetObj;

}// test examplevar o1 = {

arr: [1, 2, 3],

obj: {

key: ’value’

},

func: function(){

return 1;

}

 

1.本站遵循行业规范,任何转载的稿件都会明确标注作者和来源;2.本站的原创文章,请转载时务必注明文章作者和来源,不尊重原创的行为我们将追究责任;3.作者投稿可能会经我们编辑修改或补充。

相关文章
  • Asp.Net MVC Identity 2.2.1 使用技巧(八)

    Asp.Net MVC Identity 2.2.1 使用技巧(八)

    2017-03-01 18:01

  • JSON数据传输,解决中文乱码

    JSON数据传输,解决中文乱码

    2017-03-01 12:00

  • Webpack入门教程十一

    Webpack入门教程十一

    2017-02-27 11:01

  • 【Unity技巧】开发技巧(技巧篇)

    【Unity技巧】开发技巧(技巧篇)

    2017-02-22 14:00

网友点评
p