区别:
- 浅克隆只是克隆一层,更深层次对象级别的只拷贝引用。
- 深克隆拷贝多层,每一级别的数据都会拷贝。
克隆只针对对象、数组、函数等复杂数据
浅克隆就是将栈内存中的引用复制一份,赋给一个新的变量,本质上两个指向堆内存中的同一地址,内容也相同,其中一个变化另一个内容也会变化(根本上改变的是同一个对象)。
深克隆就是创建一个新的空对象,开辟一块内存,然后将原对象中的数据全部复制过去,完全切断两个对象间的联系。
原生js实现浅克隆
// 浅克隆
var obj ={
id:1,
name:'jack',
msg:{
age: 18
}
};
var o = {};
// 遍历obj
for(var k in obj){
o[k] = obj[k];
}
console.log(o);
o.msg.age = 20;
console.log(obj);
这里简单定义了一个对象obj,里面有两个属性,再把obj赋给变量o,输入o后发现内容是obj的内容。这里创建出的对象被放在堆内存中,而obj,o两个引用都是在栈内存中。这里涉及到一些内存位置的内容想要把对象obj拷贝给对象o,obj包含了两个属性,id和name则仅仅是一个地址,指向了obj。这里就是一个简单的浅克隆的例子,把对象obj的引用复制一份给o,让两个变量都指向obj。这时候任意一个引用对obj做出的修改都会折射到另一个引用上。
ES6中新增了浅克隆的语法糖:Object.assign(target,…sources)
var obj ={
id:1,
name:'jack'
};
var o = {};
console.log('-------------------');
Object.assign(o, obj);
console.log(o);
而深克隆本质上是创造一个完全一样的对象。
递归完成深克隆
var obj ={
id:1,
name:'jack',
msg:{
age: 18
}
};
var o = {};
// 封装函数
function deepCopy(newobj, oldobj){
for(var k in oldobj){
// 判断属性值属于哪种数据类型
// 获取属性值
var item = oldobj[k];
// 判断这个值是否是数组
if(item instanceof Array){
newobj[k] = [];
deepCopy(newobj[k], item);
}else if(item instanceof Array) {
// 判断这个值是否是对象
newobj[k] = [];
deepCopy(newobj[k], item);
}else{
// 这个值是简单数据类型
newobj[k] = item;
}
}
}
deepCopy(o, obj);
console.log(o);
总结
- 如果一个对象中的属性都是原始类型的值,不再包含对象、数组等引用类型,我们可以用浅克隆进行复制。
- 如果一个对象包含内嵌的对象或者是数组的话,我们必须需要使用深克隆对它进行深层复制。