ES6中的super关键字详解

时间:2025-01-18 22:48:39

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

首先抛出一个概念: super这个关键字,既可以当作函数使用,又可以当作对象使用

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

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

class A {}

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

注意:虽然super代表父类的构造函数,但此时返回的时B的实例,即super内部的this指的是B的实例,因此super()相当于 (this)

class A {
  constructor() {
    console.log(new.target.name);
  }
}
class B extends A {
  constructor() {
    super();
  }
}
new A() // A
new B() // B

上述代码中,指向当前正在执行的函数,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作为函数时,代表父类的构造方法,作为对象时,指向父类的原型对象,即,所以()相当于()

这里还需要注意,由于 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就可以访问

class A {}
A.prototype.x = 2;

class B extends A {
  constructor() {
    super();
    console.log(super.x) // 2
  }
}

let b = new B();

上面代码中,属性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

上面代码中,()虽然调用的是(),但是内部的this指向子类B的实例,所以应该输出2,实际执行的是(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当作普通对象来使用,代表,所以为undefined
    console.log(this.x); // 3
  }
}

let b = new B();

上面代码中, = 3这里相当于 = 3,给子类实例赋值,而取值的时候相当于,返回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

上述代码中静态方法()中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]

完!