实现Object深拷贝的方法及其原理

深拷贝的三种方法

(一)利用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": ["看电影","打游戏","看书"]}

优点:因为该方法简单、易用、有效,针对大多数常见对象类型都能进行深拷贝,且能够适用于大多数业务场景,所以该方法是最常用、最有效的方法。

缺点还比较明显:

  1. 无法处理函数和undefined等特殊类型;
  2. 当处理包含大量数据或复杂结构的对象时,可能会比较慢。

 比如如下代码:

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:

obj1的结构

通过deepCopy方法复制obj1,并输出到obj2中,obj2的结构中没有undefined和function,且原型链中丢掉了age属性:

obj2的结构

通过对比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中,深拷贝和浅拷贝是两种不同的拷贝方法,拷贝的方式会影响目标对象和源对象之间的关系。

  1. 浅拷贝:浅拷贝只复制了对象的引用,而不是复制对象本身。当改变拷贝后的对象时,原对象也会受到影响。

  2. 深拷贝:深拷贝会创建一个原对象的完全独立的副本,包括它的所有属性和嵌套对象。拷贝后的对象和原对象之间没有任何关系,对其中一个的修改不会影响另一个。

下面是深拷贝和浅拷贝的一些示例代码:

浅拷贝示例:

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,因为它们共享同一个引用。

如果需要实现深拷贝,即拷贝对象及其所有嵌套属性的值,可以使用其他方法或者自定义函数来实现。

  • 17
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值