箭头函数

[TOC]

箭头函数之前不常用的语法

直接返回一个对象

let func = (value, num) => ({ data: value + num });
func(1, 2); // { data :3 }

与变量解构

let func = ({ value, num }) => ({ data: value + num });
console.log(func({ value: 1, num: 2 })); // { data :3 }

与变量解构的应用(在react中)

一般设置state的方式


state = {
  data: {
    age: 17
  }
};

handleSet = () => {
  this.setState({
    data: this.state.data.set("key", "value")
  });
};

使用箭头函数与变量解构后可以简化为


handleSet2 = () => {
  this.setState(({ data }) => ({
    data: data.set("key", "value")
  }));
};

没有this

  • 箭头函数需要查找作用域链来确定this的值

  • 这就意味着如果箭头函数被非箭头函数包含,this绑定的就是最近一层非箭头函数的this

  • 下面这个例子,箭头函数中的this绑定的就是他的外层函数demo()的this,而demo()是在全局作用域中调用的,所以这个this, 指向的是全局作用域window

var value = 2;
function demo() {
  const value = 1;
  const b = () => {
    console.log(this);
  };
  b();
}
demo(); // 2

因为箭头函数没有this,所以不能使用call(), apply(), bind()这些方法改变this的指向

var name = "hong";
const person = {
  name: "ming"
};
function sayName() {
  return this.name;
}
const sayName2 = () => {
  return this.name;
};
console.log(sayName());             // "hong"
console.log(sayName.call(person));  // "ming"
console.log(sayName2());            // "hong" => this始终指向window 
console.log(sayName2.call(person)); // "hong" => this始终指向window 

箭头函数没有this,给我们带来了很多方便,比如之前我们希望函数内部的闭包可以拿到他的包含作用的this, 就需要提前将this保存在一个闭包可以放到的变量里面;
在匿名函数返回之前,我们将sayName()的this(指向obj)保存在变量that中,这样,我们就可以通过that访问到obj的name属性

function Person() {
  var that = this;
  that.age = 0;

  setTimeout(function() {
    console.log(that.age); // |this| 正确地指向 p 实例
  }, 1000);
}

var p = new Person();  // 0

但是由于箭头根本没有this,所以可以直接访问包含函数this

function Person() {
  this.age = 0;

  setTimeout(function() {
    console.log(this.age); // |this| 正确地指向 p 实例
  }, 1000);
}

var p = new Person(); // 0

this和arguments也存在着同样的问题。如果不使用箭头函数,但是想访问作用域中的arguments对象,必须将对该对象的引用保存到另一个闭包能够访问的变量中。

没有arguments

箭头函数没有自己的arguments对象,这不一定是一件坏事,因为这样,箭头函数就可以访问外围函数的arguments对象

如果你非要使用arguments对象怎么办呢

const demo = (...args) => {
  return args;
};
demo(1, 2, 3);  // [1, 2, 3] 

在函数柯里化的时候就很方便了,正常情况下我们需要吧外层函数的除了第一个参数外的参数与内部的参数拼接起来,如下

const add = function(a, b, c, d) {
  return a + b + c + d;
};
function curry(fn) {
  const args = Array.prototype.slice.call(arguments, 1);
  return function() {
-   const innerArgs = Array.prototype.slice.call(arguments);
    const finalArgs = [...args, ...innerArgs];
    return fn.apply(null, finalArgs);
  };
}
console.log(curry(add, 1)(2, 3, 4));  // 10

由于箭头函数没有arguments对象的特性,我们就可以直接访问他的包含函数的arguments对象,函数柯里化我们可以写成这个样子

function curry(fn) {
  return (..._args) => {
    const finalArgs = [...[].slice.call(arguments, 1), ..._args];
    return fn.apply(null, finalArgs);
  };
}
console.log(curry(add, 1)(2, 3, 4));  // 10

没有new.target属性(即不能通过new关键字调用)

普通的函数调用中(和作为构造函数来调用相对),new.target 的值是undefined;而在构造函数中调用时,new.target指向被new调用的构造函数这使得你可以检测一个函数是否是作为构造函数通过new被调用的

function Demo() {
  console.log(new.target === window.Demo);
}
new Demo();  // true
  • 这里冴羽的博客说函数有[[call]]和[[constructor]], 当通过new 调用函数时,执行 [[Construct]] 方法,创建一个实例对象,然后再执行函数体,将 this 绑定到实例上; 当直接调用的时候,执行 [[Call]] 方法,直接执行函数体。
  • 箭头函数没有[[Construct]]方法,所以不能被用作构造函数,所以通过new的方式调用会报错
  • 但是我没有看到书上有介绍过 [[Call]] 方法,等我找到回来改

把箭头当作构造函数,使用new创建的实例报错

const Demo = () => {console.log(1)};
const demo1 = new Demo(); // TypeError: Demo is not a constructor

没有原型

箭头函数不存在prototype这个属性

const DemoA = () => {
  console.log(1);
};

const DemoB = function() {
  console.log(1);
};
console.log(DemoA.prototype.constructor === DemoA); // TypeError: Cannot read property 'constructor' of undefined
console.log(DemoB.prototype.constructor === DemoB); // true

避免使用箭头函数的三种情况

由于箭头函数没有绑定this, 所以以下三种情况避免使用

  1. 对象的方法
// bad
const person = {
  name: "小红",
  sayName: () => {
    console.log(this);  // Window
    return this.name
  }
};
person.sayName();
person.sayName();   // undefined

  1. 定义原型方法
// right
function FooA() {
  this.value = 1;
}
FooA.prototype.getValue = function() {
  console.log(this);  // 指向FooA的实例
  console.log(this.value);
};

let foo1 = new FooA()
let foo2 = new FooA()
foo2.value = 2

foo1.getValue() // 1
foo2.getValue() // 2

// bad
function Foo() {
  this.value = 1
}

Foo.prototype.getValue = () => {
  console.log(this) // Window
  console.log(this.value)
}

let foo = new Foo()
foo.getValue();  // undefined
  1. 作为事件的回调函数
// bad
const button = document.getElementById('myButton');
button.addEventListener('click', () => {
    console.log(this === window); // => true
    this.innerHTML = 'Clicked button';
});

参考

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容