欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

ES6之super关键字详解

程序员文章站 2022-04-07 18:03:09
ES6重新实现了类的继承,而在继承的过程中,super关键字实现了至关重要的作用,可以说理解不了super关键字,也就掌握不了类的继承,今天我们就一起来盘盘super这个关键字首先抛出一个概念: super这个关键字,既可以当作函数使用,又可以当作对象使用 第一种情况:super作为函数时,代表父类的构造函数 ES6要求,子类的构造函数,必须执行一次super函数class A {}class B extends A { constructor() { super();//子类....

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]

完!


本文地址:https://blog.csdn.net/u013448372/article/details/111997556

相关标签: js es6 javascript