定义ES6类方法的差异

原文请戳这里Differences in Defining ES6 Class Methods

在ES6中,有两种普遍的方式用来定义class methods,但他们的行为完全不同。

第一种方式是将class methods定义为标准的类函数。

注意:这种方式,方法定义在类的原型

class A {
    foo() {  
      console.log('foo from A')
    }
  }

但是你也可以把它们定义为实例的具体函数

注意:这种定义方式不是ES6标准语法,但被transform-class-properties支持,因此广为流行。

注意:这种定义方式,方法其实是定义在构造函数内的,即直接定义在实例上的,而不是构造函数的原型上。

class A {
    foo = () => {
        console.log('foo from A')
    }
}

如果你知道ES6的箭头函数那么第一个区别很明显:它将函数调用的this变量绑定到了函数定义范围中的this变量。因此使用第二种方式定义,this被绑定到了当前的类的实例对象上。

我们可以不使用箭头函数,使用标准的class method也能实现相同的功能,只是需要在构造函数内进行额外的绑定。

class A { 
    constructor() {
      this.foo = this.foo.bind(this)
    }

    foo() {
        console.log('foo from A')
    }
}

您现在可能会认为以下类定义具有相同的行为:


0_3jvz23ydPSbkA2r5.png

然而,当我们使用javascript的类继承的时候,两者的行为完全不同。

类继承产生的不同结果:

让我们运行下面的代码:

class A { 
  constructor() {
      this.foo = this.foo.bind(this)
  }

  foo() {
      console.log('foo from A')
  }
}

class B extends A {
    foo() {
      super.foo();
      console.log('foo from B')
    }
}
new B().foo()

预期结果如下:

foo from A
foo from B

现在我们尝试使用箭头函数:

class A { 
    foo = () => {
      console.log('foo from A')
    }
}

class B extends A {
    foo = () => {
        super.foo();
        console.log('foo from B')
    }
}
new B().foo()

我们得到错误:

Cannot read property 'call' of undefined
0_EIbiqVDV59cHNOSg.png

这里发生了什么?

无论何时JavaScript中,当你调用了super.someFunction时,都会在__.proto__中查找someFunction关键词——这就是JavaScript类继承的实现方式。之所以我们在这得到了一个错误,
是因为foo不在new B()实例对象的__proto__上。

这告诉我们,不能在子类中使用箭头函数。箭头函数总是只能通过类实例属性来定义。那么,我们的classtransform-class-properties实际上都做了些什么:当具体的class instance被创建时,首先会在构造函数内附加函数。

 //第二种方式定义方法的本质是在构造函数内附加一个箭头函数,因此this会自动绑定了实例对象。
class A {
    constructor() {
          this.foo = () => {
          console.log('foo from A')
        }
     }
}  

当然,这样做,foo不会出现在__proto__且使得foo在扩展的对象中不可用。如果你需要一个方法在派生类中可以被调用(同时你希望使用ES6的class语法),可以使用class method的简写形式定义。

class A { 
  constructor() {
      this.foo = this.foo.bind(this)
  }

  foo() {
      console.log('foo from A')
  }
}

class B extends A {
    foo() {
         super.foo();
        console.log('foo from B')
    }
}
new B().foo()

最初发表于 cmichel.io

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,559评论 18 139
  • class的基本用法 概述 JavaScript语言的传统方法是通过构造函数,定义并生成新对象。下面是一个例子: ...
    呼呼哥阅读 4,062评论 3 11
  • 官方中文版原文链接 感谢社区中各位的大力支持,译者再次奉上一点点福利:阿里云产品券,享受所有官网优惠,并抽取幸运大...
    HetfieldJoe阅读 2,983评论 4 14
  • 特别说明,为便于查阅,文章转自https://github.com/getify/You-Dont-Know-JS...
    杀破狼real阅读 1,115评论 0 4
  • 最近近乎饥渴的看了很多文字,公众号推送的文章,手机里的《且听风吟》,kindle里《银河系漫游指南》,家里钢琴上的...
    阿雅喵阅读 259评论 0 0