在 JavaScript 中如何克隆对象?

做者:Luigi Nori
译者:前端小智
来源:stackabuse

有梦想,有干货,微信搜索 【大迁世界】 关注这个在凌晨还在刷碗的刷碗智。javascript

本文 GitHub https://github.com/qq449245884/xiaozhi 已收录,有一线大厂面试完整考点、资料以及个人系列文章。前端

当咱们想要复制原始值和引用值(对象)时,它们的行为会大不相同。java

原始值

咱们假设一个变量 name 具备一个与之关联的原始值(number,string,boolean,undefined 和null)。 若是咱们将此变量 name 复制到另外一个变量name2 ,则原始变量的任何修改都不会影响到第二个变量,由于它们是原始值。git

let name="前端小智";
let name2= name;
console.log (name, name2); // 前端小智, 前端小智
name="王大冶";
console.log (name,name2); // 王大冶 前端小智

引用值

可是,若是咱们对引用类型的值进行相同的操做,则咱们对一个变量所作的任何更改也将反映在另外一个变量中,由于两个变量都指向同一对象。github

数组

要拷贝数组,slice()方法用于建立数组的新副本。 能够独立修改此副本,而不会影响原始数组。面试

若是未传递任何参数,则它会精确复制数组,但数字也能够做为参数传递。 若是仅传递一个数字,它将肯定咱们要从其进行复制的索引的值,而若是传递两个数字,则将标记开始和结束。数组

// 示例1
const names = ['前端小智', '王大冶', '小力'];
const names2 = names;
console.log(names, names2);
// ["前端小智", "王大冶", "小力"] 
// ["前端小智", "王大冶", "小力"] 

// 示例2
names2[2] = '前端小力';
console.log(names, names2);
//  ["前端小智", "王大冶", "前端小力"] 
//  ["前端小智", "王大冶", "前端小力"] 

// 示例3
const name2 = names.slice();
names[2] = '我是隔壁老智';
console.log(name2, names2)
// ["前端小智", "王大冶", "前端小力"]
//  ["前端小智", "王大冶", "我是隔壁老智"]

对象

当引用值是一个对象时,也会发生一样的状况,对其属性之一的任何修改都会影响这两个变量。 若要克隆对象,请使用 Object.assign() 方法,该方法会将一个或多个源对象的全部可枚举属性的值复制到目标对象,可是此方法仅对对象的一个浅拷贝。微信

// 示例1
const names = {
  name: '前端小智',
  surname: '隔壁老智'
}

const names2 = names;
console.log(names, names2) // 打印结果是如出一辙的

// 示例2
names2.surname ='隔壁老王';
console.log(names, names2)

// {name: "前端小智", surname: "隔壁老王"}
// {name: "前端小智", surname: "隔壁老王"}

// 示例3
const names3 = Object.assign({}, names);
names3.surname = '隔壁老色P';
console.log(names, names3)

// {name: "前端小智", surname: "隔壁老王"}
// {name: "前端小智", surname: "隔壁老色P"}

要对对象进行深拷贝,须要使用其余方法。函数

正如咱们所说,Object.assign()方法只是一个浅拷贝(即,当咱们的对象没有其余对象做为属性时)才有效。 在这些状况下,必须对对象进行深拷贝。工具

与浅拷贝不一样,深拷贝以递归方式复制每一个子对象,直到全部涉及的对象都被复制为止。

咱们可使用什么方法复制对象的深层副本?

JSON.parse(JSON.stringify(obj))

此方法使用JSON.stringify()将对象转换为字符串,而后再用JSON.parse()将其转换回对象。 此方法对简单对象有效,但若是对象属性是函数时无效。

const names = {
  name: '前端小智',
  surname: '隔壁老智',
  social: {
    wx: '大迁世界',
    url: 'www.lsp.com'
  }
}
const names2 = JSON.parse(JSON.stringify(names));
names2.social.url = 'www.baidu.com';
console.log(names, names2);

/** 
{
  name: "前端小智"
  social: {wx: "大迁世界", url: "www.lsp.com"}
  surname: "隔壁老智"
}
*/

/** 
{
  name: "前端小智"
  social: {wx: "大迁世界", url: "www.baidu.com"}
  surname: "隔壁老智"
}
*/

深度拷贝

另外一种很是有趣和优雅的对象深度复制方法是使用递归函数。

咱们建立了一个deepClone(object)函数,将想要克隆的对象做为参数传递给它。在函数内部,将建立一个局部变量克隆,这是一个空对象,其中将从起始对象克隆的每一个属性都将添加到该对象中。

具体思路:

  • 若是该属性不是对象,则将其简单地克隆并添加到新的克隆对象中。
  • 若是属性是对象,则再次执行deepClone(value)函数,并将属性的值(在这种状况下为对象)做为参数传递,并重复相同的过程。
function deepClone(object) {
  var clone = {};
  for (var key in object) {
    var value = object[key];
    if (typeof(value) !== 'object') {
      clone[key] = value;
    } else {
      clone[key]=deepClone(value);
    }
  }
  return clone;
} 

deepClone({value1:1,value2:{value3:2}});
//{value1:1,value2:{value3:2}}
deepClone({value1:1,value2:{value3:{value3b:3}}});
//{value1:1,value2:{value3:{value3b:3}}}

代码部署后可能存在的BUG无法实时知道,过后为了解决这些BUG,花了大量的时间进行log 调试,这边顺便给你们推荐一个好用的BUG监控工具 Fundebug

原文:https://www.ma-o.org/en/progr...

交流

有梦想,有干货,微信搜索 【大迁世界】 关注这个在凌晨还在刷碗的刷碗智。

本文 GitHub https://github.com/qq44924588... 已收录,有一线大厂面试完整考点、资料以及个人系列文章。

相关文章
相关标签/搜索