从开始接触es6到在项目中使用已经有一段时间了,es6有不少优秀的新特性,其中最有价值的特性之一就是箭头函数,他简洁的语法以及更好理解的this值都很是的吸引我。可是新事物也是有两面性的,箭头函数有他的便捷有他的优势,可是他也有缺点,他的优势是代码简洁,this提早定义,但他的缺点也是这些,好比代码太过简洁,致使很差阅读,this提早定义,致使没法使用js进行一些es5里面看起来很是正常的操做。针对这些缺点,下面我就总结一下什么状况下不应使用箭头函数。es6
先来看下面这段代码dom
var obj = { array: [1, 2, 3], sum: () => { console.log(this === window); // => true return this.array.reduce((result, item) => result + item); } }; // Throws "TypeError: Cannot read property 'reduce' of undefined" obj.sum();
sum方法定义在obj对象上,当调用的时候咱们发现抛出了一个TypeError,由于函数中的this是window对象,因此this.array也就是undefined。缘由也很简单,相信只要了解过es6 箭头函数的都知道函数
箭头函数没有它本身的this值,箭头函数内的this值继承自外围做用域this
解决方法也很简单,就是不用呗。这里能够用es6里函数表达式的简洁语法,在这种状况下,this值就取决于函数的调用方式了。es5
var obj = { array: [1, 2, 3], sum() { console.log(this === obj); // => true return this.array.reduce((result, item) => result + item); } }; obj.sum(); // => 6
经过object.method()语法调用的方法使用非箭头函数定义,这些函数须要从调用者的做用域中获取一个有意义的this值。prototype
在对象原型上定义函数也是遵循着同样的规则code
function Person (pName) { this.pName = pName; } Person.prototype.sayName = () => { console.log(this === window); // => true return this.pName; } var person = new Person('wdg'); person.sayName(); // => undefined
使用function函数表达式对象
function Person (pName) { this.pName = pName; } Person.prototype.sayName = function () { console.log(this === person); // => true return this.pName; } var person = new Person('wdg'); person.sayName(); // => wdg
因此给对象原型挂载方法时,使用function函数表达式继承
this是js中很是强大的特色,他让函数能够根据其调用方式动态的改变上下文,而后箭头函数直接在声明时就绑定了this对象,因此再也不是动态的。
在客户端,在dom元素上绑定事件监听函数是很是广泛的行为,在dom事件被触发时,回调函数中的this指向该dom,可当咱们使用箭头函数时:事件
var button = document.getElementById('myButton'); button.addEventListener('click', () => { console.log(this === window); // => true this.innerHTML = 'Clicked button'; });
由于这个回调的箭头函数是在全局上下文中被定义的,因此他的this是window。因此当this是由目标对象决定时,咱们应该使用函数表达式:
var button = document.getElementById('myButton'); button.addEventListener('click', function() { console.log(this === button); // => true this.innerHTML = 'Clicked button'; });
在构造函数中,this指向新建立的对象实例
this instanceOf MyFunction === true
须要注意的是,构造函数不能使用箭头函数,若是这样作会抛出异常
var Person = (name) => { this.name = name; } // Uncaught TypeError: Person is not a constructor var person = new Person('wdg');
理论上来讲也是不能这么作的,由于箭头函数在建立时this对象就绑定了,更不会指向对象实例。
箭头函数可让语句写的很是的简洁,可是一个真实的项目,通常由多个开发者共同协做完成,就算由单人完成,后期也并不必定是同一我的维护,箭头函数有时候并不会让人很好的理解,好比
let multiply = (a, b) => b === undefined ? b => a * b : a * b; let double = multiply(2); double(3); // => 6 multiply(2, 3); // =>6
这个函数的做用就是当只有一个参数a时,返回接受一个参数b返回a*b的函数,接收两个参数时直接返回乘积,这个函数能够很好的工做而且看起很简洁,可是从第一眼看去并非很好理解。
为了让这个函数更好的让人理解,咱们能够为这个箭头函数加一对花括号,并加上return语句,或者直接使用函数表达式:
function multiply(a, b) { if (b === undefined) { return function (b) { return a * b; } } return a * b; } let double = multiply(2); double(3); // => 6 multiply(2, 3); // => 6
毫无疑问,箭头函数带来了不少便利。恰当的使用箭头函数可让咱们避免使用早期的.bind()函数或者须要固定上下文的地方而且让代码更加简洁。
箭头函数也有一些不便利的地方。咱们在须要动态上下文的地方不能使用箭头函数:定义须要动态上下文的函数,构造函数,须要this对象做为目标的回调函数以及用箭头函数难以理解的语句。在其余状况下,请尽情的使用箭头函数。