箭头函数

箭头函数没有自己的this,arguments,super或new.target。箭头函数适用于那些本来需要匿名函数的地方,并且它不能用作构造函数。

1.基础语法

1 (param1, param2, ..., paramN) => { statements }
1 (param1, param2, ..., paramN) => expression
2 //相当于: (param1, param2, ..., paramN) => {return expression}
3 //当函数体内只有一句代码时,retrun 和 {}可以省略
4 
5 //当只有一个参数时,圆括号也是可选的, 可省略
6 (singleParam) => {statements}
7   singleParam => {statements}
8 //没有参数的函数应该写成一对圆括号
9 () => {statements}
  //空的箭头函数返回undefined
10 let empty = () => {};

1.1.返回对象时要加括号

1 params => ({foo: bar}) //返回对象字面量表达式

1.2.支持剩余参数和默认参数

1 (param1, param2, ...rest) => {statements}
2 (param1=defaultValue1, param2, ..., paramN=defaultValueN) => {statements}

1.3.支持参数列表解构

1 let f = ([a, b] = [1, 2], {x: c} = {x : a + b}) => a + b + c;
2 f(); //6

2.箭头函数不绑定this

普通函数是根据它是如何被调用的来定义这个函数的this值

  • 如果该函数是一个构造函数,this指向一个新的对象(实例对象)
  • 在严格模式下的函数调用,this指向undefined
  • 如果该函数是一个对象的方法,则它指向这个对象

2.1箭头函数不会创建自己的this,它只会从自己的作用域链的上一层继承this。

1 function Person() {
2     this.age = 0;
3 
4     setInterval(() => {
5         this.age++;  // this 指向p 实例
6     }, 1000);
7 }
8 
9 var p = new Person();

2.2通过 call 或 apply 调用时

由于箭头函数没有自己的 this,通过 call() 或 apply() 方法调用一个函数时,只能传递参数(不能绑定this),它们的第一个参数会被忽略。(这种现象对bind方法也成立)

 

 1 var adder = {
 2     base: 1,
 3     
 4     add: function(a) {
 5         var f = v => v + this.base;
 6         return f(a);
 7     },
 8     
 9     addThruCall: function(a) {
10         var f = v => v + this.base;
11         var b = {
12             base: 2
13         };
14          return f.call(b, a);
15     }
16 };
17 
18 console.log(adder.add(1));  //2
19 console.log(adder.addThruCall(1))  // 2

3.不绑定arguments

箭头函数不绑定Arguments对象。不可以使用arguments对象,该对象在函数体内不存在。

1 function foo(n) {
2    var f = () => arguments[0] + n; //隐式绑定 foo 函数的 arguments 对象。arguments[0] 是 n, 即传给foo函数的第一个参数
3    return f();
4 }
5 
6 foo(1); //2
7 foo(2); //4
8 foo(3); //6
9 foo(3,2); //6

 在大多数情况下,使用剩余参数是相较使用arguments对象的更好选择。

 1 function foo(arg) {
 2     var f = (..args) => args[0];
 3     return f(arg);
 4 }
 5 
 6 foo(1);  //1
 7 
 8 function foo(arg1, arg2) {
 9     var f = (...args) => args[1];
10     return f(arg1, arg2);
11 }
12 
13 foo(1, 2);  //2

4.箭头函数不能用作构造函数,和new一起使用会抛出错误。

5.箭头函数没有prototype属性

6.箭头函数不能用作函数生成器,因为yield关键字通常不能再箭头函数中使用(除非是嵌套在允许使用的函数内)。

7.箭头函数不适合用作对象的方法

 1 'use strict';
 2 var obj = {
 3     i: 10,
 4     b: () => console.log(this.i, this),
 5     c: function() {
 6         console.log(this.i, this);
 7     }
 8 }
 9 
10 obj.b(); //undefined, window{...}
11 obj.c(); //10, Object{...}
12 
13 
14 var obj = {
15     a: 10
16 };
17 
18 Object.defineProperty(obj, "b", {
19     get: () => {
20         console.log(this.a, typeof this.a, this);
21         return this.a + 10;
22         //this 代表全局对象 'window', 因此,'this.a' 返回 'undefined'
23     }
24 });
25 obj.b;  //undefined "undefined" Window{postMessage: f, blur: f, focus: ƒ, close: ƒ, frames: Window, …}

8.箭头函数的解析顺序

箭头函数具有与常规函数不同的特殊运算符优先级解析规则

1 let callback;
2 
3 callback = callback || function() {};  //ok
4 
5 callback = callback || () => {}; //SyntaxError: invalid arrow-function arguments
6 
7 callback = callback || (() => {});  //ok

9.箭头函数使用三元运算符:

1 var simple = a => a > 15 ? 15 : a;
2 
3 simple(16); //15
4 simple(10); //10
5 
6 
7 let max = (a, b) => a > b ? a : b;

10.箭头函数内定义的变量及作用域

 1 //常规写法
 2 var greeting = () => {let now = new Date(); return ("Good" + ((now.getHours() > 17) ? "evening." : "day."));}
 3 greeting();          //"Good day."
 4 console.log(now);    // ReferenceError: now is not defined 标准的let作用域
 5 
 6 //参数括号内定义的变量是局部变量(默认参数)
 7 var greeting = (now=new Date()) = > "Good" + (now.getHours() > 17 ? "evening." : "day.);
 8 greeting();          //"Good day."
 9 console.log(now);    // ReferenceError: now is not defined
10 
11 //对比: 函数体内{}不使用var定义的变量是全局变量
12 var greeting = () => {now = new Date(); return ("Good" + ((now.getHours() > 17) ? " evening." : " day."));}
13 greeting();           //"Good day."
14 console.log(now);     // Fri Dec 22 2017 10:01:00 GMT+0800 (中国标准时间)
15 
16 //对比: 函数体内{} 用var定义的变量是局部变量
17 var greeting = () => {var now = new Date(); return ("Good" + ((now.getHours() > 17) ? " evening." : " day."));}
18 greeting(); //"Good day."
19 console.log(now);    // ReferenceError: now is not defined

11.箭头函数可以使用闭包

 1 //标准的闭包函数
 2 function A() {
 3     var i = 0;
 4     return function b() {
 5         return (++i);
 6     };
 7 }
 8 
 9 var v =A();
10 v();  //1
11 v(); //2
12 
13 //箭头函数的闭包(i=0 是默认参数)
14 var Add = (i=0) => {return (() => (++i))};
15 var v = Add();
16 v(); //1
17 v(); //2

12.箭头函数递归

1 var fact = (x) => (x == 0 ? 1 : x*fact(x-1));
2 
3 fact(5);  //120

 

posted @ 2020-09-17 19:09  焱雨  阅读(1109)  评论(0编辑  收藏  举报