ES6中的super关键字详解

  • Post author:
  • Post category:其他


ES6重新实现了类的继承,而在继承的过程中,super关键字实现了至关重要的作用,可以说理解不了super关键字,也就掌握不了类的继承,今天我们就一起来盘盘super这个关键字

首先抛出一个概念:



super这个关键字,既可以当作函数使用,又可以当作对象使用




第一种情况:super作为函数时,代表父类的构造函数

ES6要求,子类的构造函数,必须执行一次

super

函数

class A {}

class B extends A {
  constructor() {
    super();//子类的构造函数,必须执行一次super函数,代表父类的构造函数
  }
}

注意:虽然super代表父类的构造函数,但此时返回的时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作为函数使用时,必须出现在子类的构造函数

constructor

中,否则会报错

class A {}

class B extends A {
  m() {
    super(); // 报错
  }
}



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

class A {
  p() {
    return 2;
  }
}

class B extends A {
  constructor() {
    super();//父类的构造函数
    console.log(super.p()); // 2
  }
}

let b = new B();

上面代码中,


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();

上面代码中,属性x是定义在父类的原型对象上,所以可以使用

super.x

来访问


ES6规定,在子类普通方法中,通过

super

调用父类的方法时,

方法内部的this指向当前的子类实例

class A {
  constructor() {
    this.x = 1;
  }
  print() {
    console.log(this.x);
  }
}

class B extends A {
  constructor() {
    super();
    this.x = 2;
  }
  m() {
    super.print(); //普通方法中,通过super调用父类的方法时,方法内部的this指向当前类的实例
  }
}

let b = new B();
b.m() // 2

上面代码中,

super.print()

虽然调用的是

A.prototype.print()

,但是内部的this指向子类B的实例,所以应该输出2,实际执行的是

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; // 使用super对属性赋值,这时super相当于this
    console.log(super.x); // undefined 这个相当于在普通函数中把super当作普通对象来使用,代表A.prototype,所以为undefined
    console.log(this.x); // 3
  }
}

let b = new B();

上面代码中,

super.x = 3

这里相当于

this.x = 3

,给子类实例赋值,而取值的时候

super.x

相当于

A.prototype.x

,返回

undefined






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);//super在静态方法中指向父类
  }

  myMethod(msg) {
    super.myMethod(msg);//super在普通方法中指向父类原型
  }
}

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();//this 指向子类,而不是实例
  }
}

B.x = 3;
B.m() // 3

上述代码中静态方法B.m()中super指向父类,this指向子类,而不是子类实例


使用

super

的时候,必须显示地指出是函数还是对象,否则会报错

class A {}

class B extends A {
  constructor() {
    super();
    console.log(super); // 报错
  }
}

上面代码中的

super

,无法看出是对象还是函数,则会报错,只要可以清晰的表明是函数还是对象,就不会报错

class A {}

class B extends A {
  constructor() {
    super();
    console.log(super.valueOf() instanceof B); // true
  }
}

let b = new B();

上面代码可以看出

super

是一个对象,则不会报错

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

super

关键字

var obj = {
  toString() {
    return "MyObject: " + super.toString();
  }
};

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

完!




版权声明:本文为u013448372原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。