JavaScript箭头函数(Arrow Function)

ES6标准新增了一种新的函数:Arrow Function(箭头函数)html

更简洁的语法

咱们先来按常规语法定义函数:express

function (x) {
    return x * x;
}

该函数使用箭头函数可使用仅仅一行代码搞定!app

x => x * x

箭头函数至关于匿名函数,而且简化了函数定义。函数

箭头函数有两种格式:学习

  • 一种像上面的,只包含一个表达式,连{ ... }return都省略掉了。
  • 还有一种能够包含多条语句,这时候就不能省略{ ... }return。
x => {
    if (x > 0) {
        return x * x;
    }
    else {
        return - x * x;
    }
}

咱们能够看一下箭头函数的语法规则:this

(parameters) => { statements }

本文上面的例子中,函数有一个参数。若是函数有多个参数或者没有参数,可按照下面写法:spa

若是返回值仅仅只有一个表达式(expression), 还能够省略大括号。rest

// 无参数:
() => 3.14

//一个参数
//若是只有一个参数,能够省略括号:
x=>x*x

// 两个参数:
(x, y) => x * x + y * y

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

注意:若是要返回一个对象,就要注意,若是是单表达式,这么写的话会报错:code

// SyntaxError:
x => { foo: x }

由于和函数体的 { ... } 有语法冲突,因此要改成:htm

// ok:
x => ({ foo: x })

没有局部this的绑定

箭头函数看上去是匿名函数的一种简写,但实际上,箭头函数和匿名函数有个明显的区别:箭头函数内部的this是词法做用域,由上下文肯定。箭头函数不会绑定this。 或则说箭头函数不会改变this原本的绑定。

举个例子说明:

function Counter() {
  this.num = 0;
}
var a = new Counter();

 console.log(a.num);// 0

由于使用了关键字 new 构造,Count()函数中的 this 绑定到一个新的对象,而且赋值给a。经过 console.log 打印 a.num,会输出0。

若是咱们想每过一秒将 a.num 的值加1,该如何实现呢?可使用 setInterval() 函数。

function Counter() {
  this.num = 0;
  this.timer = setInterval(function add() {
    this.num++;
    console.log(this.num);
  }, 1000);
}

var b = new Counter();
// NaN
// NaN
// NaN
// ...

每隔一秒都会有一个NaN打印出来,而不是累加的数字。

缘由:首先函数 setInterval 没有被某个声明的对象调用,也没有使用 new 关键字,再之没有使用 bind ,  call 和 apply 。 setInterval 只是一个普通的函数。实际上 setInterval 里面的 this 绑定到全局对象的。

之因此打印 NaN ,是由于 this.num 绑定到 window 对象的 num ,而 window.num 未定义。

解决办法:使用箭头函数!使用箭头函数就不会致使 this 被绑定到全局对象。

function Counter() {
  this.num = 0;
  this.timer = setInterval(() => {
    this.num++;
    console.log(this.num);
  }, 1000);
}
var b = new Counter();
// 1
// 2
// 3
// ...

经过 Counter 构造函数绑定的 this 将会被保留。在 setInterval 函数中, this 依然指向咱们新建立的b对象。

再举个例子:

因为JavaScript函数对 this 绑定的错误处理,下面的例子没法获得预期结果:

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 :

var obj = {
    birth: 1990,
    getAge: function () {
        var b = this.birth; // 1990
        var fn = () => new Date().getFullYear() - this.birth; // this指向obj对象
        return fn();
    }
};
obj.getAge(); // 25

若是使用箭头函数,之前的那种hack写法:

var that = this;

就再也不须要了。

因为this在箭头函数中已经按照词法做用域绑定了,因此,用 call() 或者 apply() 调用箭头函数时,没法对 this 进行绑定,即传入的第一个参数被忽略:

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

 

本文部份内容借鉴自:JavaScript学习笔记(十二)——箭头函数(Arrow Function)

 

亲,若是您感受本文有用,请点个赞再走吧✌(>‿◠)!!

相关文章
相关标签/搜索