前言
最近學習了慕課網上的vue仿餓了嗎的視頻,體會到了es6的箭頭函數的妙用,因此特別總結了一下箭頭函數的用法。
首先看一下es6的箭頭函數語法
<pre>
([param] [, param]) => {
statements
}
param => expression
</pre>
aram 是參數,根據參數個數不同,分這幾種情況:
1, () => { … } // 零個參數用或多個參數 () 表示且()不能省略;
2, x => { … } // 一個參數可以省略 ();
看第一種情況
<pre>
var f = () => 5;
// 等同于
var f = function () { return 5 };
var sum = (num1, num2) => num1 + num2;
// 等同于
var sum = function(num1, num2) {
return num1 + num2;
};
</pre>
第二種情況
<pre>
var f = v => v;
上面箭頭函數等價于
var f = function(v) {
return v;
};
</pre>
當然=>后面也不一定非得接return 之后的語句,看下面:
<pre>
// ES5
$("#confetti-btn").click(function (event) {
playTrumpet();
fireConfettiCannon();
});
// ES6
$("#confetti-btn").click(event => {
playTrumpet();
fireConfettiCannon();
});</pre>
但是需要注意的是,多行語句需要用{}括起來,單行表達式不需要{},并且會作為函數返回值。
和普通函數一樣,箭頭函數也可以使用剩余參數和 默認參數。
<pre>
var func1 = (x = 1, y = 2) => x + y;
func1(); // 得到 3
var func2 = (x, ...args) => { console.log(args) };
func2(1,2,3); // 輸出 [2, 3] </pre>
使用注意點
函數體內的this對象,就是定義時所在的對象,而不是使用時所在的對象。
不可以當作構造函數,也就是說,不可以使用new命令,否則會拋出一個錯誤。
不可以使用arguments對象,該對象在函數體內不存在。如果要用,可以用Rest參數代替。
不可以使用yield命令,因此箭頭函數不能用作Generator函數。
this指向的固定化,并不是因為箭頭函數內部有綁定this的機制,實際原因是箭頭函數根本沒有自己的this,導致內部的this就是外層代碼塊的this。正是因為它沒有this,所以也就不能用作構造函數。
箭頭函數內部沒有constructor方法,也沒有prototype,所以不支持new操作。但是它對this的處理與一般的普通函數不一樣。箭頭函數的 this 始終指向函數定義時的 this,而非執行時。我們通過一個例子來理解:
var o = {
x : 1,
func : function() { console.log(this.x) },
test : function() {
setTimeout(function() {
this.func();
}, 100);
}
};
o.test(); // TypeError : this.func is not a function
上面的代碼會出現錯誤,因為this的指向從o變為了全局(函數調用中的this都是指向全局的)。如果大家對JavaScript中的this不是很熟悉的話,可以看看我寫過的一篇文章,深入理解javascript之this。好了,回歸正題,我們需要修改上面的代碼如下:
var o = {
x : 1,
func : function() { console.log(this.x) },
test : function() {
var _this = this;
setTimeout(function() {
_this.func();
}, 100);
}
};
o.test();
通過使用外部事先保存的this就行了。這里就可以利用到箭頭函數了,我們剛才說過,箭頭函數的 this 始終指向函數定義時的 this,而非執行時。所以我們將上面的代碼修改如下:
var o = {
x : 1,
func : function() { console.log(this.x) },
test : function() {
setTimeout(() => { this.func() }, 100);
}
};
o.test();
我們還需要注意一點的就是這個this是不會改變指向對象的,我們知道call和apply可以改變this的指向,但是在箭頭函數中是無效的。
var x = 1,
o = {
x : 10,
test : () => this.x
};
o.test(); // 1
o.test.call(o); // 依然是1