开启掘金成长之旅!这是我参与「掘金日新计划 · 12 月更文挑战」的第4天,点击查看活动详情 昨天和一个朋友聊天,我问她是怎么理解深浅拷贝的,她的回答是:浅拷贝就是拷贝另一个对象只会拷贝这个对象的地址值,其中一个修改另一个也会变;深拷贝就是在拷贝对象的时候,将这个对象里的所有引用结构都拷贝一份,内存中会存在两个数据结构完全相同又相互独立的对象。由此,今天我想聊一下深浅拷贝:
我以前是这么理解的
说到深浅拷贝,需要明确一点:
引用类型,进行赋值时,赋值的是地址
let obj1 = {
name: 'zs',
age: 18,
car: {
brand: '宝马',
price: 100
}
}
let obj2 = {
...obj1
}
console.log(obj2); // 浅拷贝
let obj3 = JSON.parse(JSON.stringify(obj1))
console.log(obj3); // 深拷贝
深拷贝和浅拷贝只针对像 Object, Array 这样的复杂对象。它们最根本的区别在于是否真正获取了一个对象的复制实体,而不是引用。 简单来说,假设B复制了A,当修改B时,看A是否会发生变化,如果A变了,说明是浅拷贝;如果A没变,那就是深拷贝
浅拷贝直接“=”赋值或者“...”即可
深拷贝的话常用JSON.stringify把对象转成字符串,再用JSON.parse把字符串转成新的对象。
现在的理解
首先,深浅拷贝,当然是针对于引用类型的,对象之间直接赋值,赋值的是地址,这些肯定是不会有错的。所以,直接赋值的对象之间,会相互影响也是肯定的了,比如,A = B,那么当修改B时,A肯定会发生变化,大家注意到没有,这是不是和上面所描述的浅拷贝很像?
所以啊,之前我所理解的浅拷贝肯定是大错特错了,因为既然直接赋值就是这样的效果,那还要浅拷贝这个东西存在干嘛,它存在的意义在哪呢?
真正的浅拷贝,应该是将这个对象完全拷贝了一份,创造了一个新的对象,这个新对象也有自己的地址,但是需要注意,如果原对象只有一层,也就是它没有子对象,那么原对象和新对象之间肯定是不会相互影响的;可是如果它确实有子对象呢,那么在拷贝的时候,只是将这个子对象的地址拷贝了一份给新对象,也就是说,原对象和新对象中的这个子对象是相同的地址,那么毫无疑问,无论改变哪个对象的这个子对象,都会影响到另一个对象中的子对象,这才是浅拷贝的正解。
至于说深拷贝,简单来说,就是深层次的浅拷贝,相当于是递归浅拷贝,不论对象的层级有多深,子对象有多少层,都会将每一个子对象的内容拷贝出来并创建一个新地址,最终形成的新对象与原对象之间,内容完全相同,但地址完全不同,包括其子对象的地址也是完全不同,彼此之间不会有任何影响。
另外,深浅拷贝常用的方法,浅拷贝最常用的自然是扩展运算符...或者Object.assign()方法,深拷贝的话就JSON.parse(JSON.stringify())
end
希望从我个人理解的变化,能够帮助到大家对深浅拷贝的理解更加深入,感谢!