数据类型


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

在 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

2024/08/22/1724302679612.webp

浅拷贝与深拷贝的理解


浅拷贝


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

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

拷贝


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

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

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


浅拷贝


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 修改为了 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