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]
完!