理解JS中的浅拷贝与深拷贝

1,058 阅读5分钟

数据类型

在进入正题之前,先说个数据类型

在JS中,数据类型分两种:基本数据类型、引用数据类型

基本数据类型

JS中的基本数据类型有五种: nullundefinedbooleanstringnumber
数据变量是直接按值存放的,他们的值在内存中占据着固定大小的空间,并被保存在栈内存中,可以直接访问,并且是简单的数据段,其数据类型的值是不可变的。

var str = "xxx";
str[0] = "y";
console.log(str);    // xxx

基本类型的比较是值的比较
只要它们的值相等就认为他们是相等的,例如:

var a = 1;
var b = 1;
console.log(a === b);	// true

引用数据类型

引用数据类型包括对象和数组,其存储在堆当中,而变量实际上是一个存放在栈内存的指针,这个指针指向堆内存中的地址。 当我们访问的时候,实际上是访问指针,然后指针去寻找对象或数组,其数据类型的值是可变的。

var str = [1,2,3];
str[0] = 4;
console.log(str);   // [4,2,3]

引用类型的比较是引用的比较
我们对 js 中的引用类型进行操作的时候,都是操作其对象的引用(保存在栈内存中的指针),所以比较两个引用类型,是看其的引用是否指向同一个对象。例如:

var a = [1,2,3];
var b = [1,2,3];
console.log(a === b);  // false

浅拷贝与深拷贝的理解

浅拷贝

只复制指向某个对象的指针,而不复制对象本身,新旧对象共享一块内存;

简单的说,浅拷贝就是将一个对象的内存地址的“”编号“”复制给另一个对象。即在真正访问的时候还是会访问到被复制对象。 或者只是深拷贝了第一层的引用类型,而没有拷贝更深层次的应用类型,而是利用复制地址的方式,这也是浅拷贝。

深拷贝

复制并创建一个一模一样的对象,不共享内存,修改新对象,旧对象保持不变。

简单的说,深拷贝就是先新建一个空对象,内存中新开辟一块地址,把被复制对象的所有可枚举的(注意可枚举的对象)属性方法一一复制过来,注意要用递归来复制子对象里面的所有属性和方法,直到子子.....属性为基本数据类型。
关键点:开辟新内存、递归复制。

浅拷贝与深拷贝的实现方式

浅拷贝

var str1 = {
    name:"Fan"
}
var str2 = str1
str2.name = "Jun"
console.log(str1.name); // Jun

这里首先创建了一个 str1 对象,然后将str1复制给了 str2, 但是这里仅仅是指针的复制,所以在修改 str2.name 的时候,实际上是修改的同一个堆中的对象,既浅拷贝。


var str = {
    a:1,
    b:{
        d:"Fan"
    },
    c:[1,2,3]
}
function Test(obj){
    var newStr = {};
    for (var item in obj){
        newStr[item] = obj[item];
    }
    return newStr;
}
var newStr = Test(str);
console.log(newStr.b.d === str.b.d);    // true

这段代码是通过for in的形式将对象进行复制,这里可以看到复制只是对于指针的复制,得到的新的对象还是指向同一个堆中的对象,所以是浅拷贝。


var str1 = {
    name: 'Fan', 
    age: 22,
    other: {
        school: 'HuangHuai'
    }
}
var str2 = Object.assign({}, str1);
str2.name = 'Jun'
console.log(str1.name)  // Fan

str2.other.school = 'ShiYan'
console.log(str1.other.school)  //ShiYan

只从表面上来看,似乎Object.assign()的目标对象是{ },是一个新的对象(开辟了一块新的内存空间),是深拷贝。

当我们修改str2.name的时候,str1.name没有改变,但是当我们修改 str2.other.school 的时候,str1.other.school 同样也发生了变化。

Object.assign()也是浅拷贝,或者说只是深拷贝了第一层,这样我们认为它还是浅拷贝。


var a = [1, [2, 3, 4], {
    name: 'Fan'
}];
var b = a.concat(5)

a[0] = 6;
console.log(b[0]) // 1 看起来像深拷贝

a[1][0] = 999;
console.log(b[1][0]) // 999 浅拷贝

a[2].name = 'Fan'
console.log(b[2].name) // Fan 浅拷贝

可以看到通过concat返回的新数组,只有改变其中一个的布尔值、字符串、数值,另一个不会改变,但是改变其中的对象、数组时,可以发现,另一个也在同时改变,即还是引用原来的堆中的内容。


var a = [1, [2, 3, 4], {
    name: 'Fan'
}];
var b = a.slice(0)

a[0] = 6;
console.log(b[0]) // 1 看起来像深拷贝

a[1][0] = 999;
console.log(b[1][0]) // 999 浅拷贝

a[2].name = 'Fan'
console.log(b[2].name) // Fan 浅拷贝

这段代码仅仅是将上一段中的concat修改为了slice,发现结果也是一样的,即slice方法得到的也是浅拷贝。

深拷贝

JSON.stringify() 和 JSON.parse()

var str1 = {
    name: 'Fan', 
    age: 22,
    other: {
        school: 'HuangHuai'
    }
}
var str2 = JSON.parse(JSON.stringify(str1));
str2.name = 'Jun'
console.log(str1.name) // Fan

str2.other.school = 'ShiYan'
console.log(str1.other.school) // HuangHuai

可以看出通过JSON.stringify先将对象转化为字符串,然后再通过JSON.parse()转化为对象,这个对象就是完全在开辟的新的内存空间中的对象 。

虽然这种方式可以实现深拷贝,但是会存在不足,就是他只能拷贝符合JSON格式的数据,如果不是JSON格式的数据,则不行.
例如:

let obj = {name:"Fan",age:function(){}}
let newObj = JSON.parse(JSON.stringify(obj))
console.log(newObj)  // { name: 'Fan' }

使用递归实现深拷贝

function deepClone(source) {
    const targetObj = source.constructor === Array ? [] : {}; // 判断复制的目标是数组还是对象
    for (let 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;
}

var str1 = {
    arr: [1, 2, 3],
    obj: {
        key: 'value'
    },
    fn: function () {
        return 1;
    }
};
var str3 = deepClone(str1);

console.log(str3 === str1); // false
console.log(str3.obj === str1.obj); // false
console.log(str3.fn === str1.fn); // true

6_6