JS Arrow Function

不会前端实在是太不方便了!!!

不会前端,即使你有想法有逻辑,还是不能实现出一个完整的产品出来,这次借着看组里自动化平台前端源码的机会,入门了一下vue.js,在写demo的时候,碰到了一个=>函数的写法,了解了一下之后发现它类似Java里的lambda表达式,但在js里也有一些细节需要注意,在此记录一下。

箭头函数的概念

JS的箭头函数(lambda表达式),其主要意图是定义轻量级的内联回调函数

  • 回调函数:这个在之前的博文里写过
  • 内联函数:函数调用的过程,是调用栈入栈出栈的过程,这是比较耗时的。内联(inline)也就是内嵌,也就意味着就像C语言#define一样,当编译器发现某段代码在调用一个内联函数时,它不是去调用该函数,而是将该函数的代码,整段内嵌到当前位置。这样做的好处是省去了调用的过程,加快程序运行速度,缺点就是会消耗空间。
1
x => x * x

上面的箭头函数相当于:

1
2
3
function (x) {
return x * x;
}

不同场景箭头函数的用法

箭头函数简化了函数定义,相当于只需指定一个参数到结果的映射。箭头函数有两种格式,一种像上面的,只包含一个表达式,连{ … }和return都省略掉了。还有一种可以包含多条语句,这时候就不能省略{ … }和return:

1
2
3
4
5
6
7
8
x => {
if (x > 0) {
return x * x;
}
else {
return - x * x;
}
}

如果参数不是一个,就需要用括号()括起来:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
// 两个参数:
(x, y) => x * x + y * y

// 无参数:
() => 3.14

// 可变参数:
(x, y, ...rest) => {
var i, sum = x + y;
for (i=0; i<rest.length; i++) {
sum += rest[i];
}
return sum;
}

如果要返回一个对象,因为和函数体的{ … }有语法冲突,所以要改为:

1
x => ({ foo: x })

JS中箭头函数与匿名函数的区别(this)

在之前讲闭包的博文中,也提到了匿名函数的概念,JS中箭头函数与匿名函数最大的区别是箭头函数内部的this是词法作用域(lexically scoped),由上下文确定。个人感觉这里的词法作用域,和闭包里引用环境的概念非常像。

在匿名函数里,由于JavaScript函数对this绑定的错误处理,下面的例子无法得到预期结果,需要用点hack的方法,比如let that = this

1
2
3
4
5
6
7
8
9
10
var obj = {
birth: 1990,
getAge: function () {
var b = this.birth; // 1990
var fn = function () {
return new Date().getFullYear() - this.birth; // this指向window或undefined
};
return fn();
}
};

现在,箭头函数完全修复了this的指向,this总是指向词法作用域,也就是外层调用者obj:

1
2
3
4
5
6
7
8
9
var obj = {
birth: 1990,
getAge: function (year) {
var b = this.birth; // 1990
var fn = (y) => y - this.birth; // this.birth仍是1990
return fn.call({birth:2000}, year);
}
};
obj.getAge(2015); // 25