前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >深入理解class的继承,助你成为优秀的前端

深入理解class的继承,助你成为优秀的前端

原创
作者头像
前端老鸟
修改2019-07-29 18:35:45
3120
修改2019-07-29 18:35:45
举报

Class 的继承

简介

Class 可以通过extends关键字实现继承,这比 ES5 的通过修改原型链实现继承,要清晰和方便很多。

  class Point {

  }



  class ColorPoint extends Point {

  }

上面代码定义了一个ColorPoint类,该类通过extends关键字,继承了Point类的所有属性和方法。但是由于没有部署任何代码,所以这两个类完全一样,等于复制了一个Point类。下面,我们在ColorPoint内部加上代码。

class ColorPoint extends Point {

  constructor(x, y, color) {

    super(x, y); // 调用父类的constructor(x, y)

    this.color = color;

  }



  toString() {

    return this.color + ' ' + super.toString(); // 调用父类的toString()

  }

}

上面代码中,constructor方法和toString方法之中,都出现了super关键字,它在这里表示父类的构造函数,用来新建父类的this对象。

子类必须在constructor方法中调用super方法,否则新建实例时会报错。这是因为子类自己的this对象,必须先通过父类的构造函数完成塑造,得到与父类同样的实例属性和方法,然后再对其进行加工,加上子类自己的实例属性和方法。如果不调用super方法,子类就得不到this对象。

class Point { /\* ... \*/ }



class ColorPoint extends Point {

  constructor() {

  }

}



let cp = new ColorPoint(); // ReferenceErro

上面代码中,ColorPoint继承了父类Point,但是它的构造函数没有调用super方法,导致新建实例时报错。

ES5 的继承,实质是先创造子类的实例对象this,然后再将父类的方法添加到this上面(Parent.apply(this))。ES6 的继承机制完全不同,实质是先将父类实例对象的属性和方法,加到this上面(所以必须先调用super方法),然后再用子类的构造函数修改this。

如果子类没有定义constructor方法,这个方法会被默认添加,代码如下。也就是说,不管有没有显式定义,任何一个子类都有constructor方法。

class ColorPoint extends Point {

}



// 等同于

class ColorPoint extends Point {

  constructor(...args) {

    super(...args);

  }

}

另一个需要注意的地方是,在子类的构造函数中,只有调用super之后,才可以使用this关键字,否则会报错。这是因为子类实例的构建,基于父类实例,只有super方法才能调用父类实例。

class Point {

  constructor(x, y) {

    this.x = x;

    this.y = y;

  }

}



class ColorPoint extends Point {

  constructor(x, y, color) {

    this.color = color; // ReferenceErro

    super(x, y);

    this.color = color; // 正确

  }

}

上面代码中,子类的constructor方法没有调用super之前,就使用this关键字,结果报错,而放在super方法之后就是正确的。

下面是生成子类实例的代码。

let cp = new ColorPoint(25, 8, 'green');



cp instanceof ColorPoint // true

cp instanceof Point // true

上面代码中,实例对象cp同时是ColorPoint和Point两个类的实例,这与 ES5 的行为完全一致。

最后,父类的静态方法,也会被子类继承。

class A {

  static hello() {

    console.log('hello world');

  }

}



class B extends A {

}



B.hello()  // hello world

上面代码中,hello()是A类的静态方法,B继承A,也继承了A的静态方法。

super 关键字

super这个关键字,既可以当作函数使用,也可以当作对象使用。在这两种情况下,它的用法完全不同。

第一种情况,super作为函数调用时,代表父类的构造函数。ES6 要求,子类的构造函数必须执行一次super函数。

class A {}



class B extends A {

  constructor() {

    super();

  }

}

上面代码中,子类B的构造函数之中的super(),代表调用父类的构造函数。这是必须的,否则 JavaScript 引擎会报错。

注意,super虽然代表了父类A的构造函数,但是返回的是子类B的实例,即super内部的this指的是B的实例,因此super()在这里相当于A.prototype.constructor.call(this)。

class A {

  constructor() {

    console.log(new.target.name);

  }

}

class B extends A {

  constructor() {

    super();

  }

}

new A() // A

new B() // B

上面代码中,new.target指向当前正在执行的函数。可以看到,在super()执行时,它指向的是子类B的构造函数,而不是父类A的构造函数。也就是说,super()内部的this指向的是B。

作为函数时,super()只能用在子类的构造函数之中,用在其他地方就会报错。

class A {}



class B extends A {

  m() {

    super(); // 报错

  }

}

上面代码中,super()用在B类的m方法之中,就会造成语法错误。

第二种情况,super作为对象时,在普通方法中,指向父类的原型对象;在静态方法中,指向父类。

class A {

  p() {

    return 2;

  }

}



class B extends A {

  constructor() {

    super();

    console.log(super.p()); // 2

  }

}



let b = new B();

上面代码中,子类B当中的super.p(),就是将super当作一个对象使用。这时,super在普通方法之中,指向A.prototype,所以super.p()就相当于A.prototype.p()。

这里需要注意,由于super指向父类的原型对象,所以定义在父类实例上的方法或属性,是无法通过super调用的。

class A {

  constructor() {

    this.p = 2;

  }

}



class B extends A {

  get m() {

    return super.p;

  }

}



let b = new B();

b.m // undefined

上面代码中,p是父类A实例的属性,super.p就引用不到它。

如果属性定义在父类的原型对象上,super就可以取到。

class A {}

A.prototype.x = 2;



class B extends A {

  constructor() {

    super();

    console.log(super.x) // 2

  }

}



let b = new B();
class A {

  constructor() {

    this.x = 1;

  }

  print() {

    console.log(this.x);

  }

}



class B extends A {

  constructor() {

    super();

    this.x = 2;

  }

  m() {

    super.print();

  }

}



let b = new B();

b.m() // 2

上面代码中,super.print()虽然调用的是A.prototype.print(),但是A.prototype.print()内部的this指向子类B的实例,导致输出的是2,而不是1。也就是说,实际上执行的是super.print.call(this)。

由于this指向子类实例,所以如果通过super对某个属性赋值,这时super就是this,赋值的属性会变成子类实例的属性。

class A {

  constructor() {

    this.x = 1;

  }

}



class B extends A {

  constructor() {

    super();

    this.x = 2;

    super.x = 3;

    console.log(super.x); // undefined

    console.log(this.x); // 3

  }

}



let b = new B();

上面代码中,super.x赋值为3,这时等同于对this.x赋值为3。而当读取super.x的时候,读的是A.prototype.x,所以返回undefined。

如果super作为对象,用在静态方法之中,这时super将指向父类,而不是父类的原型对象。

class Parent {

  static myMethod(msg) {

    console.log('static', msg);

  }



  myMethod(msg) {

    console.log('instance', msg);

  }

}



class Child extends Parent {

  static myMethod(msg) {

    super.myMethod(msg);

  }



  myMethod(msg) {

    super.myMethod(msg);

  }

}



Child.myMethod(1); // static 1



var child = new Child();

child.myMethod(2); // instance 2

上面代码中,super在静态方法之中指向父类,在普通方法之中指向父类的原型对象。

另外,在子类的静态方法中通过super调用父类的方法时,方法内部的this指向当前的子类,而不是子类的实例。

class A {

  constructor() {

    this.x = 1;

  }

  static print() {

    console.log(this.x);

  }

}



class B extends A {

  constructor() {

    super();

    this.x = 2;

  }

  static m() {

    super.print();

  }

}



B.x = 3;

B.m() // 3

上面代码中,静态方法B.m里面,super.print指向父类的静态方法。这个方法里面的this指向的是B,而不是B的实例。

注意,使用super的时候,必须显式指定是作为函数、还是作为对象使用,否则会报错。

class A {}



class B extends A {

  constructor() {

    super();

    console.log(super); // 报错

  }

}

上面代码中,console.log(super)当中的super,无法看出是作为函数使用,还是作为对象使用,所以 JavaScript

class A {}



class B extends A {

  constructor() {

    super();

    console.log(super.valueOf() instanceof B); // true

  }

}



let b = new B();

上面代码中,super.valueOf()表明super是一个对象,因此就不会报错。同时,由于super使得this指向B的实例,所以super.valueOf()返回的是一个B的实例。

最后,由于对象总是继承其他对象的,所以可以在任意一个对象中,使用super关键字。

var obj = {

  toString() {

    return "MyObject: " + super.toString();

  }

};



obj.toString(); // MyObject: [object Object]

类的 prototype 属性和__proto__属性

大多数浏览器的 ES5 实现之中,每一个对象都有__proto__属性,指向对应的构造函数的prototype属性。Class 作为构造函数的语法糖,同时有prototype属性和__proto__属性,因此同时存在两条继承链。

(1)子类的__proto__属性,表示构造函数的继承,总是指向父类。

(2)子类prototype属性的__proto__属性,表示方法的继承,总是指向父类的prototype属性。

class A {

}



class B extends A {

}



B.\_\_proto\_\_ === A // true

B.prototype.\_\_proto\_\_ === A.prototype // true

上面代码中,子类B的__proto__属性指向父类A,子类B的prototype属性的__proto__属性指向父类A的prototype属性。

这样的结果是因为,类的继承是按照下面的模式实现的。

class A {

}



class B {

}



// B 的实例继承 A 的实例

Object.setPrototypeOf(B.prototype, A.prototype);



// B 继承 A 的静态属性

Object.setPrototypeOf(B, A);



const b = new B();

《对象的扩展》一章给出过Object.setPrototypeOf方法的实现。



Object.setPrototypeOf = function (obj, proto) {

  obj.\_\_proto\_\_ = proto;

  return obj;

}

因此,就得到了上面的结果。

Object.setPrototypeOf(B.prototype, A.prototype);

// 等同于

B.prototype.\_\_proto\_\_ = A.prototype;



Object.setPrototypeOf(B, A);

// 等同于

B.\_\_proto\_\_ = A;

这两条继承链,可以这样理解:作为一个对象,子类(B)的原型(\_\_proto\_\_属性)是父类(A);作为一个构造函数,子类(B)的原型对象(prototype属性)是父类的原型对象(prototype属性)的实例。



B.prototype = Object.create(A.prototype);

// 等同于

B.prototype.\_\_proto\_\_ = A.prototype;

extends关键字后面可以跟多种类型的值。



class B extends A {

}

上面代码的A,只要是一个有prototype属性的函数,就能被B继承。由于函数都有prototype属性(除了Function.prototype函数),因此A可以是任意函数。

下面,讨论两种情况。第一种,子类继承Object类。

class A extends Object {

}



A.\_\_proto\_\_ === Object // true

A.prototype.\_\_proto\_\_ === Object.prototype // true

这种情况下,A其实就是构造函数Object的复制,A的实例就是Object的实例。

第二种情况,不存在任何继承。

class A {

}



A.\_\_proto\_\_ === Function.prototype // true

A.prototype.\_\_proto\_\_ === Object.prototype // true

这种情况下,A作为一个基类(即不存在任何继承),就是一个普通函数,所以直接继承Function.prototype。但是,A调用后返回一个空对象(即Object实例),所以A.prototype.__proto__指向构造函数(Object)的prototype属性。

实例的 __proto__ 属性

子类实例的__proto__属性的__proto__属性,指向父类实例的__proto__属性。也就是说,子类的原型的原型,是父类的原型。

var p1 = new Point(2, 3);

var p2 = new ColorPoint(2, 3, 'red');



p2.\_\_proto\_\_ === p1.\_\_proto\_\_ // false

p2.\_\_proto\_\_.\_\_proto\_\_ === p1.\_\_proto\_\_ // true

上面代码中,ColorPoint继承了Point,导致前者原型的原型是后者的原型。



因此,通过子类实例的\_\_proto\_\_.\_\_proto\_\_属性,可以修改父类实例的行为。



p2.\_\_proto\_\_.\_\_proto\_\_.printName = function () {

  console.log('Ha');

};



p1.printName() // "Ha"

上面代码在ColorPoint的实例p2上向Point类添加方法,结果影响到了Point的实例p1。

以上就是对class继承的初步分析,希望大家可以多多指出有误的地方

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Class 的继承
    • 简介
      • super 关键字
        • 类的 prototype 属性和__proto__属性
          • 实例的 __proto__ 属性
          领券
          问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档