美文网首页
JavaScript复制对象

JavaScript复制对象

作者: shujuan0618 | 来源:发表于2019-10-31 14:54 被阅读0次

在JavaScript这门语言中,数据类型分为两大类:基本数据类型和复杂数据类型。基本数据类型包括Number、Boolean、String、Null、String、Symbol(ES6 新增),而复杂数据类型包括Object,而所有其他引用类型(Array、Date、RegExp、Function、基本包装类型(Boolean、String、Number)、Math等)都是Object类型的实例对象,因此都可以继承Object原型对象的一些属性和方法。
而对于基本数据类型来说,复制一个变量值,本质上就是copy了这个变量。一个变量值的修改,不会影响到另外一个变量。看一个简单的例子。
let val = 123;
let copy = val;
console.log(copy); //123
val = 456; //修改val的值对copy的值不产生影响
console.log(copy); //123
而对于复杂数据类型来说,同基本数据类型实现的不太相同。对于复杂数据类型的复制,要注意的是,变量名只是指向这个对象的指针。当我们将保存对象的一个变量赋值给另一个变量时,实际上复制的是这个指针,而两个变量都指向都一个对象。因此,一个对象的修改,会影响到另外一个对象。
// obj只是指向对象的指针
let obj = {
character: 'peaceful'
};
//copy变量复制了这个指针,指向同一个对象
let copy = obj;
console.log(copy); //{character: 'peaceful'}
obj.character = 'lovely';
console.log(copy); //{character: 'lovely'}
有一副很形象的图描述了复杂数据类型复制的原理

同理,在复制一个数组时,变量名只是指向这个数组对象的指针;在复制一个函数时,函数名只是指向这个函数对象的指针
let arr = [1, 2, 3];
let copy = arr;
console.log(copy); // [1, 2, 3]
arr[0] = 'keith';
console.log(copy); // 数组对象被改变: ['keith', 2, 3]
arr = null;
console.log(copy); // ['keith', 2, 3] 即使arr=null,也不会影响copy。因此此时的arr变量只是一个指向数组对象的指针

function foo () {
return 'hello world';
};
let bar = foo;
console.log(foo());
foo = null; //foo只是指向函数对象的指针
console.log(bar());
因此,我们应该如何实现对象的深浅复制?
复制对象
在JavaScript中,复制对象分为两种方式,浅复制和深复制。
浅复制没有办法去真正的去复制一个对象,而只是保存了对该对象的引用;而深复制可以实现真正的复制一个对象。
浅复制
在ES6中,Object对象新增了一个assign方法,可以实现对象的浅复制。这里谈谈Object.assign方法的具体用法,因为稍后会分析jQuery的extend方法,实现的原理同Object.assign方法差不多
Object.assign的第一个参数是目标对象,可以跟一或多个源对象作为参数,将源对象的所有可枚举([[emuerable]] === true)复制到目标对象。这种复制属于浅复制,复制对象时只是包含对该对象的引用。Object.assign(target, [source1, source2, ...])
? 如果目标对象与源对象有同名属性,则后面的属性会覆盖前面的属性
? 如果只有一个参数,则直接返回该参数。即Object.assign(obj) === obj
? 如果第一个参数不是对象,而是基本数据类型(Null、Undefined除外),则会调用对应的基本包装类型
? 如果第一个参数是Null和Undefined,则会报错;如果Null和Undefined不是位于第一个参数,则会略过该参数的复制
要实现对象的浅复制,可以使用Object.assign方法
let target = {a: 123};
let source1 = {b: 456};
let source2 = {c: 789};
let obj = Object.assign(target, source1, source2);
console.log(obj);
不过对于深复制来说,Object.assign方法无法实现
let target = {a: 123};
let source1 = {b: 456};
let source2 = {c: 789, d: {e: 'lovely'}};
let obj = Object.assign(target, source1, source2);
source2.d.e = 'peaceful';
console.log(obj); // {a: 123, b: 456, c: 789, d: {e: 'peaceful'}}
从上面代码中可以看出,source2对象中e属性的改变,仍然会影响到obj对象
深复制
在实际的开发项目中,前后端进行数据传输,主要是通过JSON实现的。JSON全称:JavaScript Object Notation,JavaScript对象表示法。
JSON对象下有两个方法,一是将JS对象转换成字符串对象的JSON.stringify方法;一个是将字符串对象转换成JS对象的JSON.parse方法。
这两个方法结合使用可以实现对象的深复制。也就是说,当我们需要复制一个obj对象时,可以先调用JSON.stringify(obj),将其转换为字符串对象,然后再调用JSON.parse方法,将其转换为JS对象。就可以轻松的实现对象的深复制
let obj = {
a: 123,
b: {
c: 456,
d: {
e: 789
}
}
};
let copy = JSON.parse(JSON.stringify(obj));
// 对obj对象无论怎么修改,都不会影响到copy对象
obj.b.c = 'hello';
obj.b.d.e = 'world';
console.log(copy); // {a: 123, b: {c: 456, d: {e: 789}}}
当然,使用这种方式实现深复制有一个缺点就是必须给JSON.parse方法传入的字符串必须是合法的JSON,否则会抛出错误

来自 https://www.cnblogs.com/libin-1/p/6845458.html

相关文章

  • JavaScript复制对象

    首先,为了创建一个对象完全相同的拷贝,需要确保两件事情: 1. 拷贝必须具有与原对象相同的原型 2. 拷贝必须具有...

  • JavaScript复制对象

    在JavaScript这门语言中,数据类型分为两大类:基本数据类型和复杂数据类型。基本数据类型包括Number、B...

  • 原生JavaScript对象复制

    简单深度复制 对象合并或者称为浅复制(对象只是复制应用,原对象改变会影响新对象,新对象改变不会影响原来的对象) 高...

  • 浅拷贝与深拷贝

    前言 在 javascript 中有不同的方法来复制对象,如果你还不熟悉这门语言的话,复制对象时就会很容易掉进陷阱...

  • 浅谈对象

    1、复制对象JavaScript初学者最常见的问题之一就是如何复制一个对象。思考一下这个对象: 如何准确的表示my...

  • javascript 对象的深度复制

    在组件直接的通讯中,我们经常需要传递复杂的数据对象data,如果接收的组件直接对data进行处理,其处理结果将是发...

  • JavaScript对象的深复制

    js的数据类型可以分为两大类: 值类型(原始类型):比如String、Number类型都是值类型。 引用类型:比如...

  • 详解js中对象的深浅拷贝

    在JavaScript中,对对象进行拷贝的场景比较常见。但是简单的复制语句只能对对象进行浅拷贝,即复制的是一份引用...

  • JavaScript中的对象复制(Object Clone)

    JavaScript中并没有直接提供对象复制(Object Clone)的方法。因此下面的代码中改变对象b的时候,...

  • React 数据为什么要使用immutable方式?

    深复制与浅复制 因为JavaScript存储对象都是存地址的,所以浅复制会导致 obj 和 obj1指向同一块内存...

网友评论

      本文标题:JavaScript复制对象

      本文链接:https://www.haomeiwen.com/subject/fjltbctx.html