深拷贝与浅拷贝
2019-05-21 本文已影响0人
弹指一挥间_e5a3
什么是深拷贝与浅拷贝?
就是假设B复制了A,当修改A时,看B是否会发生变化,如果B也跟着变了,说明这是浅拷贝。如果B没变,那就是深拷贝,自食其力。
例如let a = [0,1,2,3,4],b=a复制,修改a[0]=1,此时数组b的值也发生了变化,这就是浅拷贝。深拷贝就是修改a的值,而b的值不发生变化。
image.png
为什么?
那么这里,就得引入基本数据类型与引用数据类型的概念了。
面试常问,基本数据类型有哪些,number,string,boolean,null,undefined,symbol以及未来ES10新增的BigInt(任意精度整数)七类。
引用数据类型(Object类)有常规名值对的无序对象{a:1},数组[1,2,3],以及函数等。
而这两类数据存储分别是这样的:
a.基本类型--名值存储在栈内存中,例如let a=1;
image当你b=a复制时,栈内存会新开辟一个内存,例如这样:
image所以当你此时修改a=2,对b并不会造成影响,因为此时的b已自食其力,翅膀硬了,不受a的影响了。当然,let a=1,b=a;虽然b不受a影响,但这也算不上深拷贝,因为深拷贝本身只针对较为复杂的object类型数据。
b.引用数据类型--名存在栈内存中,值存在于堆内存中,但是栈内存会提供一个引用的地址指向堆内存中的值,我们以上面浅拷贝的例子画个图:
image当b=a进行拷贝时,其实复制的是a的引用地址,而并非堆里面的值。
image而当我们a[0]=1时进行数组修改时,由于a与b指向的是同一个地址,所以自然b也受了影响,这就是所谓的浅拷贝了。
image那,要是在堆内存中也开辟一个新的内存专门为b存放值,就像基本类型那样,岂不就达到深拷贝的效果了
image实现深拷贝的三种方法
1.递归
function deepClone(obj) {
var newObj = obj instanceof Array ? [] : {};
//obj属于基本数据类型,直接返回obj
if(typeof obj !== 'object') {
return obj;
} else {
//obj属于数组或对象,遍历它们
for(var i in obj) {
newObj[i] = typeof obj[i] === 'object' ? deepClone(obj[i]):obj[i];
}
}
return newObj;
}
2.JSON对象的parse和stringify
function deepClone(obj){
let _obj = JSON.stringify(obj),
objClone = JSON.parse(_obj);
return objClone
}
3.jQuery的extend方法。
let a=[0,1,[2,3],4],
b=$.extend(true,[],a);