深拷贝的三种方法
(一)利用JSON进行深拷贝【最常用】
使用 JSON.stringify() 和 JSON.parse():这种方式可以将对象转换为字符串,再将字符串转换回对象。通过这种方式可以实现深拷贝,但是有一些限制,例如不能拷贝函数、循环引用等。
function deepCopy(obj) {
return JSON.parse(JSON.stringify(obj));
}
var obj1 = {name:'懂点哲学的程序员',age:30,hobbies:['看电影','打游戏']};
var obj2 = deepCopy(obj1);
obj2.hobbies.push('看书');
console.log(obj1);
//输出:{"name": "懂点哲学的程序员","age": 30,"hobbies": ["看电影","打游戏"]}
console.log(obj2);
//输出:{"name": "懂点哲学的程序员","age": 30,"hobbies": ["看电影","打游戏","看书"]}
优点:因为该方法简单、易用、有效,针对大多数常见对象类型都能进行深拷贝,且能够适用于大多数业务场景,所以该方法是最常用、最有效的方法。
缺点还比较明显:
- 无法处理函数和undefined等特殊类型;
- 当处理包含大量数据或复杂结构的对象时,可能会比较慢。
比如如下代码:
function deepCopy(obj) {
return JSON.parse(JSON.stringify(obj));
}
var obj1 = {
name: "懂点哲学的程序员",
country: undefined,
hobbies: function(){
return ['看电影','打游戏']
}
}
var obj2 = deepCopy(obj1);
console.log(obj1);
console.log(obj2);
console.log(obj1)输出obj1的结构中包含undefined和function:
![](https://img-blog.csdnimg.cn/direct/e60b589bf1e049299fcbdd2672901bac.png)
通过deepCopy方法复制obj1,并输出到obj2中,obj2的结构中没有undefined和function,且原型链中丢掉了age属性:
![](https://img-blog.csdnimg.cn/direct/8b7747b277b845fd902116c25e2f700f.png)
通过对比obj1和obj2的结构可知,通过这种方法进行的深拷贝,无法处理对象的函数和undefined类型的数据。不能处理undefined数据可能算是一个优点。
(二)递归深拷贝
在JavaScript中想要实现对象的深拷贝,还可以使用递归或循环遍历对象的属性,递归深拷贝可以处理各种对象类型,包括undefined和function以。
下面是一个使用递归方式实现对象深拷贝的代码示例:
function deepCopy(obj) {
if (typeof obj !== 'object' || obj === null) {
return obj;
}
let copy = Array.isArray(obj) ? [] : {};
for (let key in obj) {
if (obj.hasOwnProperty(key)) {
copy[key] = deepCopy(obj[key]);
}
}
return copy;
}
使用以上的deepCopy()
函数可以实现对象的深拷贝,示例如下:
var obj1 = {
name: "懂点哲学的程序员",
country: undefined,
hobbies: function(){
return ['看电影','打游戏']
}
}
var obj2 = deepCopy(obj1);
obj2.age = 30;
console.log("obj1:",obj1);
console.log("obj2:",obj2);
执行效果如下图:
可以看到,修改拷贝对象obj2
的属性不会影响原始对象obj1
的值。且函数类型和undefined类型数据,也能拷贝过来。
其缺点是:递归函数需要手动调整,对于结构复杂的对象,如果没有正确处理对象的属性,可能会导致拷贝不完全或产生错误。
切记:递归深拷贝,不能处理存在循环引用的对象(一个对象,直接或间接引用其自身),否则拷贝会陷入死循环。
(三)进行深拷贝的第三方工具
用第三方库:如果不想自己实现深拷贝的逻辑,可以使用一些第三方库,例如 lodash 的 cloneDeep 方法。
const _ = require('lodash');
let obj = {a: 1, b: {c: 2}};
let copy = _.cloneDeep(obj);
深拷贝和浅拷贝的区别
在JavaScript中,深拷贝和浅拷贝是两种不同的拷贝方法,拷贝的方式会影响目标对象和源对象之间的关系。
-
浅拷贝:浅拷贝只复制了对象的引用,而不是复制对象本身。当改变拷贝后的对象时,原对象也会受到影响。
-
深拷贝:深拷贝会创建一个原对象的完全独立的副本,包括它的所有属性和嵌套对象。拷贝后的对象和原对象之间没有任何关系,对其中一个的修改不会影响另一个。
下面是深拷贝和浅拷贝的一些示例代码:
浅拷贝示例:
var obj1 = { a: 1, b: { c: 2 } };
var obj2 = Object.assign({}, obj1);
obj2.a = 3;
obj2.b.c = 4;
console.log(obj1); // { a: 1, b: { c: 4 } }
console.log(obj2); // { a: 3, b: { c: 4 } }
在这个示例中,使用Object.assign()
方法进行浅拷贝,改变拷贝后的对象obj2
的属性,原对象obj1
的属性也会发生变化。
Object.assign是浅拷贝
Object.assign是浅拷贝的原因是它只会拷贝对象的属性的引用,而不会拷贝属性的值。当源对象的属性值是一个引用类型(如对象或数组)时,目标对象拷贝的是该引用,而不是该引用指向的具体值。这意味着如果修改了源对象属性的值,目标对象中对应属性的值也会随之改变,因为它们共享同一个引用。这就是浅拷贝的概念。
举个例子,假设有一个源对象source和一个目标对象target:
const source = {
a: {
b: 2
}
};
const target = Object.assign({}, source);
source.a.b = 3;
console.log(target.a.b); // 输出3
在上面的例子中,source和target对象都有一个属性a,该属性的值是一个对象{ b: 2 }
。当使用Object.assign方法对source进行浅拷贝赋值给target后,修改source.a.b的值为3,target.a.b的值也随之改变为3,因为它们共享同一个引用。
如果需要实现深拷贝,即拷贝对象及其所有嵌套属性的值,可以使用其他方法或者自定义函数来实现。