深拷贝与浅拷贝

concat方法与slice方法

这两个方法实际上返回的是一个浅拷贝的结果。

concat

var arr3 = [{name:'xiaobai',age:18},10];
var arr4 = [1,2,3];
arr5 = arr3.concat(arr4);
//更改arr3 中索引值为 0 的对象的属性
arr3[0].name = '小白';
console.log(arr5);//输出结果:[{name:'小白',age:18}, 10, 1, 2, 3]

slice

         var arr3 = [{name:'xiaobai',age:18},10];
         arr5 = arr3.slice();
         //更改arr3 中索引值为 0 的对象的属性
         arr3[0].name = '小白';
         console.log(arr5);//输出结果:[{name:'小白',age:18}]

从上面两个例子可以看出,当数组中存有复杂数据类型时,他们返回的是一个浅拷贝的结果。

数组/对象的深拷贝

JSON.parse(JSON.stringify())深拷贝

var obj = {
    name: 'xiaobai',
    age: 18
    }

var copy = JSON.parse(JSON.stringify(obj));
console.log(obj);//输出结果:{name: "xiaobai", age: 18}
console.log(copy);//输出结果:{name: "xiaobai", age: 18}
console.log(obj === copy);//输出结果:false

可以看到,对于复杂类型的对象,使用JSON先序列化然后再反序列化,得到的结果虽然一致,但是变量 obj 和 copy 所指向的对象不是同一个,他们之间的比较结果是false,这种情况下两个对象之一发生的任何变化都不会影响到其他一个。此时我们就实现了复杂类型的深拷贝了!但是千万注意!JSON.Stringtify这个方法不能深拷贝函数!!这个要记得!

  基于数组的深拷贝其实就是针对数组中的复杂类型的深拷贝,所以我们对整个数组使用JSON.parse(JSON.stringify())深拷贝即可。

递归深拷贝

  在使用for...in 来实现深拷贝的时候需要注意一个坑,即原型链上的可枚举属性会被遍历出来,这是我们所不希望看到的,所以进入正题之前首先说一下如何避免遍历到原型链上的可枚举属性:

var obj = {
    name: 'xiaohei',
    age: 18
}
//在原型链上添加一个自定义的可枚举属性
obj.__proto__.eat = 'hello';

for(var key in obj){
    console.log(key);
    //遍历结果:name    age     eat
}

可以看到,原型链上的可枚举属性也被遍历出来了,为了避免出现这样的情况,我们可以使用js提供的一个方法hasOwnProperty(),该方法会判断当前属性是自身属性还是继承而来的,或者说是自身属性还是原型链上的可枚举属性,返回值使boolean,自身属性为true,原型链可枚举属性为false。

var obj = {
    name: 'xiaohei',
    age: 18
}
//在原型链上添加一个自定义的可枚举属性
obj.__proto__.eat = 'hello';

for(var key in obj){
    if(obj.hasOwnProperty(key)){
        console.log(key);
        //输出结果: name    age
    }
}

 OK,进入正题,编写一个递归方法,用来深拷贝复杂类型:

/**
 2  * 
 3  * @param {Object} obj 传递进去进行递归的数组或者对象
 4  */
 5 function deepClone(obj){
            if(obj === null) return null;
            if(typeof obj !== 'object') return obj;
            if(obj instanceof RegExp) {
                return new RegExp(obj);
            }
            if(obj instanceof Date) {
                return new Date(obj);
            }
            /* 这里之所以使用constructor,是为了可以使的克隆出来的对象
            与被克隆的对象都是同一个类的实例对象 */
            let newObj = new obj.constructor;
            for(let key in obj) {
                if(obj.hasOwnProperty(key)) {
                    newObj[key] = deepClone(obj[key]);
                }
            }
            return newObj;
        }
24 
25 var arr = [1, 2, { name: 'xiaohei', age: 18 }];
26 var newArr = deepClone(arr);
27 //修改父数组中对象元素的属性值
28 arr[2].name = '小黑';
29 console.log(arr);
30 console.log(newArr);
31 console.log(arr === newArr);

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值