Object.assign()的用法 -- 用于将所有可枚举属性的值从一个或多个源对象复制到目标对象,返回目标对象

语法: Object.assign(target, …sources) target: 目标对象,sources: 源对象
用于将所有可枚举属性的值从一个或多个源对象复制到目标对象。它将返回目标对象。

 1 const target = { a: 1, b: 2 };
 2 const source = { b: 4, c: 5 };
 3 const returnedTarget = Object.assign(target, source);
 4 
 5 console.log(target); // Object { a: 1, b: 4, c: 5 }
 6 console.log(returnedTarget); // Object { a: 1, b: 4, c: 5 }

用法示例:
复制一个对象

1 const obj = { a: 1 };
2 const copy = Object.assign({}, obj);
3 console.log(copy); // { a: 1 }


深拷贝问题
针对深拷贝,需要使用其他办法,因为 Object.assign()拷贝的是属性值。假如源对象的属性值是一个对象的引用,那么它也只指向那个引用。

 1 let obj1 = { a: 0 , b: { c: 0}}; 
 2 let obj2 = Object.assign({}, obj1); 
 3 console.log(JSON.stringify(obj2)); // { a: 0, b: { c: 0}}
 4 
 5 obj1.a = 1; 
 6 console.log(JSON.stringify(obj1)); // { a: 1, b: { c: 0}} 
 7 console.log(JSON.stringify(obj2)); // { a: 0, b: { c: 0}}
 8 
 9 obj2.a = 2; 
10 console.log(JSON.stringify(obj1)); // { a: 1, b: { c: 0}} 
11 console.log(JSON.stringify(obj2)); // { a: 2, b: { c: 0}}
12 
13 obj2.b.c = 3; 
14 console.log(JSON.stringify(obj1)); // { a: 1, b: { c: 3}} 
15 console.log(JSON.stringify(obj2)); // { a: 2, b: { c: 3}}
16 
17 // 深拷贝 
18 obj1 = { a: 0 , b: { c: 0}}; 
19 let obj3 = JSON.parse(JSON.stringify(obj1)); 
20 obj1.a = 4; 
21 obj1.b.c = 4; 
22 console.log(JSON.stringify(obj3)); // { a: 0, b: { c: 0}}


合并对象

1 const o1 = { a: 1 };
2 const o2 = { b: 2 };
3 const o3 = { c: 3 };
4 
5 const obj = Object.assign(o1, o2, o3);
6 console.log(obj); // { a: 1, b: 2, c: 3 }
7 console.log(o1); // { a: 1, b: 2, c: 3 }, 注意目标对象自身也会改变。


合并具有相同属性的对象

1 const o1 = { a: 1, b: 1, c: 1 };
2 const o2 = { b: 2, c: 2 };
3 const o3 = { c: 3 };
4 
5 const obj = Object.assign({}, o1, o2, o3);
6 console.log(obj); // { a: 1, b: 2, c: 3 },注意属性被后续参数中具有相同属性的其他对象覆盖

继承属性和不可枚举属性是补课拷贝的

 1 const obj = Object.create({foo: 1}, { // foo 是个继承属性。
 2 bar: {
 3 value: 2 // bar 是个不可枚举属性。
 4 },
 5 baz: {
 6 value: 3,
 7 enumerable: true // baz 是个自身可枚举属性。
 8 }
 9 });
10 
11 const copy = Object.assign({}, obj);
12 console.log(copy); // { baz: 3 }


原始类型会包装为对象

1 const v1 = "abc";
2 const v2 = true;
3 const v3 = 10;
4 const v4 = Symbol("foo")
5 
6 const obj = Object.assign({}, v1, null, v2, undefined, v3, v4); 
7 // 原始类型会被包装,null 和 undefined 会被忽略。
8 // 注意,只有字符串的包装对象才可能有自身可枚举属性。
9 console.log(obj); // { "0": "a", "1": "b", "2": "c" }


拷贝Symbol类型的属性

1 const o1 = { a: 1 };
2 const o2 = { [Symbol('foo')]: 2 };
3 
4 const obj = Object.assign({}, o1, o2);
5 console.log(obj); // { a : 1, [Symbol("foo")]: 2 } (cf. bug 1207182 on Firefox)
6 Object.getOwnPropertySymbols(obj); // [Symbol(foo)]

异常会打断后续的拷贝

 1 const target = Object.defineProperty({}, "foo", {
 2 value: 1,
 3 writable: false
 4 }); // target 的 foo 属性是个只读属性。
 5 
 6 Object.assign(target, {bar: 2}, {foo2: 3, foo: 3, foo3: 3}, {baz: 4});
 7 // TypeError: "foo" is read-only
 8 // 注意这个异常是在拷贝第二个源对象的第二个属性时发生的。
 9 
10 console.log(target.bar); // 2,说明第一个源对象拷贝成功了。
11 console.log(target.foo2); // 3,说明第二个源对象的第一个属性也拷贝成功了。
12 console.log(target.foo); // 1,只读属性不能被覆盖,所以第二个源对象的第二个属性拷贝失败了。
13 console.log(target.foo3); // undefined,异常之后 assign 方法就退出了,第三个属性是不会被拷贝到的。
14 console.log(target.baz); // undefined,第三个源对象更是不会被拷贝到的。


拷贝访问器

 1 const obj = {
 2 foo: 1,
 3 get bar() {
 4 return 2;
 5 }
 6 };
 7 
 8 let copy = Object.assign({}, obj); 
 9 console.log(copy); // { foo: 1, bar: 2 } copy.bar的值来自obj.bar的getter函数的返回值
10 
11 // 下面这个函数会拷贝所有自有属性的属性描述符
12 function completeAssign(target, ...sources) {
13 sources.forEach(source => {
14 let descriptors = Object.keys(source).reduce((descriptors, key) => {
15 descriptors[key] = Object.getOwnPropertyDescriptor(source, key);
16 return descriptors;
17 }, {});
18 
19 // Object.assign 默认也会拷贝可枚举的Symbols
20 Object.getOwnPropertySymbols(source).forEach(sym => {
21 let descriptor = Object.getOwnPropertyDescriptor(source, sym);
22 if (descriptor.enumerable) {
23 descriptors[sym] = descriptor;
24 }
25 });
26 Object.defineProperties(target, descriptors);
27 });
28 return target;
29 }
30 
31 copy = completeAssign({}, obj);
32 console.log(copy);
33 // { foo:1, get bar() { return 2 } }

原文链接:https://blog.csdn.net/Calla_Lj/article/details/89478765

猜你喜欢

转载自www.cnblogs.com/itgezhu/p/11585891.html