1
0
mirror of https://github.com/ruanyf/es6tutorial.git synced 2025-05-24 18:32:22 +00:00

docs(object): edit object/Object.assign

This commit is contained in:
ruanyf 2017-10-08 13:47:40 +08:00
parent 84503df2cf
commit 55dc77e01d
2 changed files with 88 additions and 2 deletions

View File

@ -586,6 +586,70 @@ foo.bind({}).name // "bound foo"
(function(){}).bind({}).name // "bound "
```
## new.target
JavaScript 语言的函数可以直接调用,也可以通过`new`命令调用。
```javascript
function fn() {
this.foo = 'hello';
}
// 调用方式一
fn()
// 调用方式二
new fn()
```
以前,一般在函数内部通过`instanceof`运算符判断,函数到底是哪一种方式调用。
```javascript
function fn() {
if (this instanceof fn) { // 通过 new 命令调用
this.foo = 'hello';
} else {
throw new Error('该函数只能通过 new 调用');
}
}
// 报错
fn()
// 不报错
new fn()
```
这种方法的问题是,如果通过`call()``apply()``bind()`这些方法绑定`this`,就会判断失效。
```javascript
// 不报错
fn.call(new fn())
```
为了解决这个问题,精确判断是否通过`new`调用ES6 引入了`new.target`属性。如果通过`new`调用,`new.target`将指向当前正在执行的函数,其他情况都指向`undefined`
```javascript
function fn() {
if (new.target === fn) { // 通过 new 命令调用
this.foo = 'hello';
} else {
throw new Error('该函数只能通过 new 调用');
}
}
// 报错
fn()
// 报错
fn.call(new fn())
// 不报错
new fn()
```
注意,`new.target`只能在函数体内部使用,如果在函数体外部使用就会报错。
## 箭头函数
### 基本用法

View File

@ -441,6 +441,8 @@ Object.assign({ a: 'b' }, { [Symbol('c')]: 'd' })
### 注意点
**1浅拷贝**
`Object.assign`方法实行的是浅拷贝,而不是深拷贝。也就是说,如果源对象某个属性的值是对象,那么目标对象拷贝得到的是这个对象的引用。
```javascript
@ -453,6 +455,8 @@ obj2.a.b // 2
上面代码中,源对象`obj1``a`属性的值是一个对象,`Object.assign`拷贝得到的是这个对象的引用。这个对象的任何变化,都会反映到目标对象上面。
**2同名属性的替换**
对于这种嵌套的对象,一旦遇到同名属性,`Object.assign`的处理方法是替换,而不是添加。
```javascript
@ -464,9 +468,11 @@ Object.assign(target, source)
上面代码中,`target`对象的`a`属性被`source`对象的`a`属性整个替换掉了,而不会得到`{ a: { b: 'hello', d: 'e' } }`的结果。这通常不是开发者想要的,需要特别小心。
一些函数库提供`Object.assign`的定制版本(比如 Lodash 的`_.defaultsDeep`方法),可以解决浅拷贝的问题,得到深拷贝的合并。
一些函数库提供`Object.assign`的定制版本(比如 Lodash 的`_.defaultsDeep`方法),可以得到深拷贝的合并。
注意,`Object.assign`可以用来处理数组,但是会把数组视为对象。
**3数组的处理**
`Object.assign`可以用来处理数组,但是会把数组视为对象。
```javascript
Object.assign([1, 2, 3], [4, 5])
@ -475,6 +481,22 @@ Object.assign([1, 2, 3], [4, 5])
上面代码中,`Object.assign`把数组视为属性名为0、1、2的对象因此源数组的0号属性`4`覆盖了目标数组的0号属性`1`
**4取值函数的处理**
`Object.assign`只能进行值的复制,如果要复制的值是一个取值函数,那么将求值后再复制。
```javascript
const source = {
get foo() { return 1 }
};
const target = {};
Object.assign(target, source)
// { foo: 1 }
```
上面代码中,`source`对象的`foo`属性是一个取值函数,`Object.assign`不会复制这个取值函数,只会拿到值以后,将这个值复制过去。
### 常见用途
`Object.assign`方法有很多用处。