共 2670 字,读完需 5 分钟。编译自 Dmitri Pavlutin 的文章,对原文内容作了精简和代码风格优化。ES6 中引入的箭头函数可让咱们写出更简洁的代码,可是部分场景下使用箭头函数会带来严重的问题,有哪些场景?会致使什么问题?该怎么解决,容我慢慢道来。javascript
能见证天天在用的编程语言不断演化是一件让人很是兴奋的事情,从错误中学习、探索更好的语言实现、创造新的语言特性是推进编程语言版本迭代的动力。JS 近几年的变化就是最好的例子, 以 ES6 引入的箭头函数(arrow functions)、class 等特性为表明,把 JS 的易用性推到了新的高度。html
关于 ES6 中的箭头函数,网上有不少文章解释其做用和语法,若是你刚开始接触 ES6,能够从这里开始。任何事物都具备两面性,语言的新特性经常被误解、滥用,好比箭头函数的使用就存在不少误区。接下来,笔者会经过实例介绍该避免使用箭头函数的场景,以及在这些场景下该如何使用函数表达式(function expressions)、函数声明或者方法简写(shorthand method)来保障代码正确性和可读性。java
JS 中对象方法的定义方式是在对象上定义一个指向函数的属性,当方法被调用的时候,方法内的 this
就会指向方法所属的对象。es6
由于箭头函数的语法很简洁,可能很多同窗会忍不住用它来定义字面量方法,好比下面的例子 JS Bin:web
const calculator = {
array: [1, 2, 3],
sum: () => {
console.log(this === window); // => true
return this.array.reduce((result, item) => result + item);
}
};
console.log(this === window); // => true
// Throws "TypeError: Cannot read property 'reduce' of undefined"
calculator.sum();复制代码
calculator.sum
使用箭头函数来定义,可是调用的时候会抛出 TypeError
,由于运行时 this.array
是未定义的,调用 calculator.sum
的时候,执行上下文里面的 this
仍然指向的是 window
,缘由是箭头函数把函数上下文绑定到了 window
上,this.array
等价于 window.array
,显而后者是未定义的。express
解决的办法是,使用函数表达式或者方法简写(ES6 中已经支持)来定义方法,这样能确保 this
是在运行时是由包含它的上下文决定的,修正后的代码以下 JS Bin:编程
const calculator = {
array: [1, 2, 3],
sum() {
console.log(this === calculator); // => true
return this.array.reduce((result, item) => result + item);
}
};
calculator.sum(); // => 6复制代码
这样 calculator.sum
就变成了普通函数,执行时 this
就指向 calculator
对象,天然能获得正确的计算结果。浏览器
一样的规则适用于原型方法(prototype method)的定义,使用箭头函数会致使运行时的执行上下文错误,好比下面的例子 JS Bin:编程语言
function Cat(name) {
this.name = name;
}
Cat.prototype.sayCatName = () => {
console.log(this === window); // => true
return this.name;
};
const cat = new Cat('Mew');
cat.sayCatName(); // => undefined复制代码
使用传统的函数表达式就能解决问题 JS Bin:函数
function Cat(name) {
this.name = name;
}
Cat.prototype.sayCatName = function () {
console.log(this === cat); // => true
return this.name;
};
const cat = new Cat('Mew');
cat.sayCatName(); // => 'Mew'复制代码
sayCatName
变成普通函数以后,被调用时的执行上下文就会指向新建立的 cat
实例。
this
是 JS 中很强大的特性,能够经过多种方式改变函数执行上下文,JS 内部也有几种不一样的默认上下文指向,但普适的规则是在谁上面调用函数 this
就指向谁,这样代码理解起来也很天然,读起来就像在说,某个对象上正在发生某件事情。
可是,箭头函数在声明的时候就绑定了执行上下文,要动态改变上下文是不可能的,在须要动态上下文的时候它的弊端就凸显出来。好比在客户端编程中常见的 DOM 事件回调函数(event listenner)绑定,触发回调函数时 this
指向当前发生事件的 DOM 节点,而动态上下文这个时候就很是有用,好比下面这段代码试图使用箭头函数来做事件回调函数 JS Bin:
const button = document.getElementById('myButton');
button.addEventListener('click', () => {
console.log(this === window); // => true
this.innerHTML = 'Clicked button';
});复制代码
在全局上下文下定义的箭头函数执行时 this
会指向 window
,当单击事件发生时,浏览器会尝试用 button
做为上下文来执行事件回调函数,可是箭头函数预约义的上下文是不能被修改的,这样 this.innerHTML
就等价于 window.innerHTML
,然后者是没有任何意义的。
使用函数表达式就能够在运行时动态的改变 this
,修正后的代码 JS Bin:
const button = document.getElementById('myButton');
button.addEventListener('click', function() {
console.log(this === button); // => true
this.innerHTML = 'Clicked button';
});复制代码
当用户单击按钮时,事件回调函数中的 this
实际指向 button
,这样的 this.innerHTML = 'Clicked button'
就能按照预期修改按钮中的文字。
构造函数中的 this
指向新建立的对象,当执行 new Car()
的时候,构造函数 Car
的上下文就是新建立的对象,也就是说 this instanceof Car === true
。显然,箭头函数是不能用来作构造函数, 实际上 JS 会禁止你这么作,若是你这么作了,它就会抛出异常。
换句话说,箭头构造函数的执行并无任何意义,而且是有歧义的。好比,当咱们运行下面的代码 JS Bin:
const Message = (text) => {
this.text = text;
};
// Throws "TypeError: Message is not a constructor"
const helloMessage = new Message('Hello World!');复制代码
构造新的 Message
实例时,JS 引擎抛了错误,由于 Message
不是构造函数。在笔者看来,相比旧的 JS 引擎在出错时悄悄失败的设计,ES6 在出错时给出具体错误消息是很是不错的实践。能够经过使用函数表达式或者函数声明 来声明构造函数修复上面的例子 JS Bin:
const Message = function(text) {
this.text = text;
};
const helloMessage = new Message('Hello World!');
console.log(helloMessage.text); // => 'Hello World!'复制代码
箭头函数容许你省略参数两边的括号、函数体的花括号、甚至 return
关键词,这对编写更简短的代码很是有帮助。这让我想起大学计算机老师给学生留过的有趣做业:看谁能使用 C 语言编写出最短的函数来计算字符串的长度,这对学习和探索新语言特性是个不错的法子。可是,在实际的软件工程中,代码写完以后会被不少工程师阅读,真正的 write once, read many times
,在代码可读性方面,最短的代码可能并不老是最好的。必定程度上,压缩了太多逻辑的简短代码,阅读起来就没有那么直观,好比下面的例子 JS Bin:
const multiply = (a, b) => b === undefined ? b => a * b : a * b;
const double = multiply(2);
double(3); // => 6
multiply(2, 3); // => 6复制代码
multiply
函数会返回两个数字的乘积或者返回一个能够继续调用的固定了一个参数的函数。代码看起来很简短,但大多数人第一眼看上去可能没法当即搞清楚它干了什么,怎么让这段代码可读性更高呢?有不少办法,能够在箭头函数中加上括号、条件判断、返回语句,或者使用普通的函数 JS Bin:
function multiply(a, b) {
if (b === undefined) {
return function (b) {
return a * b;
}
}
return a * b;
}
const double = multiply(2);
double(3); // => 6
multiply(2, 3); // => 6复制代码
为了让代码可读性更高,在简短和啰嗦之间把握好平衡是很是有必要的。
箭头函数无疑是 ES6 带来的重大改进,在正确的场合使用箭头函数能让代码变的简洁、短小,但某些方面的优点在另一些方面可能就变成了劣势,在须要动态上下文的场景中使用箭头函数你要格外的当心,这些场景包括:定义对象方法、定义原型方法、定义构造函数、定义事件回调函数。
本文做者王仕军,商业转载请联系做者得到受权,非商业转载请注明出处。若是你以为本文对你有帮助,请点赞!若是对文中的内容有任何疑问,欢迎留言讨论。想知道我接下来会写些什么?欢迎订阅个人掘金专栏。