在模拟实现class继承之前,如果还不熟悉JavaScript的继承,可以看一下这篇文章JavaScript继承

    首先看一下ES6的继承

    1. // ES6继承
    2. class Animal {
    3. static footer = 'footer'
    4. constructor(opt) {
    5. this.name = opt.name
    6. }
    7. getName() {
    8. return this.name
    9. }
    10. }
    11. class Cat extends Animal {
    12. constructor(opt) {
    13. super(opt)
    14. this.age = opt.age
    15. }
    16. getAge() {
    17. return this.age
    18. }
    19. }
    20. const cat = new Cat({ name: 'miao', age: 5 })

    在上面代码中,Cat继承Animal,通过使用extends继承Animal原型上的方法,通过super调用父构造函数

    关于super:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Operators/super

    super([arguments]); // 调用 父对象/父类 的构造函数 super.functionOnParent([arguments]); // 调用 父对象/父类 上的方法

    我们要想模拟实现ES6继承,关键看ES6 class extends最终转成ES5是什么样,这里急用Babel工具:在线转换
    摘出最主要的代码

    1. function _inherits(subClass, superClass) {
    2. if (typeof superClass !== "function" && superClass !== null) {
    3. throw new TypeError("Super expression must either be null or a function");
    4. }
    5. subClass.prototype = Object.create(superClass && superClass.prototype, {
    6. constructor: { value: subClass, writable: true, configurable: true }
    7. });
    8. if (superClass) _setPrototypeOf(subClass, superClass);
    9. }
    10. // ES6继承
    11. var Animal =
    12. /*#__PURE__*/
    13. (function() {
    14. function Animal(opt) {
    15. _classCallCheck(this, Animal);
    16. this.name = opt.name;
    17. }
    18. _createClass(Animal, [
    19. {
    20. key: "getName",
    21. value: function getName() {
    22. return this.name;
    23. }
    24. }
    25. ]);
    26. return Animal;
    27. })();
    28. _defineProperty(Animal, "footer", "footer");
    29. var Cat =
    30. /*#__PURE__*/
    31. (function(_Animal) {
    32. _inherits(Cat, _Animal);
    33. function Cat(opt) {
    34. var _this;
    35. _classCallCheck(this, Cat);
    36. _this = _possibleConstructorReturn(
    37. this,
    38. _getPrototypeOf(Cat).call(this, opt)
    39. );
    40. _this.age = opt.age;
    41. return _this;
    42. }
    43. _createClass(Cat, [
    44. {
    45. key: "getAge",
    46. value: function getAge() {
    47. return this.age;
    48. }
    49. }
    50. ]);
    51. return Cat;
    52. })(Animal);
    53. var cat = new Cat({
    54. name: "miao",
    55. age: 5
    56. });

    其中最主要的是 _inherits(Cat, _Animal);_getPrototypeOf(Cat).call(this, opt) ,看到这里我们发现有点像寄生组合式继承。别急,还有跟他不一样的地方,
    我们运行转换后的代码,看一下Cat.__proto__
    image.png
    发现Cat.proto是指向Animal的,在转换后的代码中也能找到对应的地方,就是在_inherits函数最后一句

    所以es6 class的继承实现就是在寄生组合式继承的基础上将子类的proto属性指向父类,为什么要这样做?
    因为寄生组合式继承没法继承父类的静态方法

    模拟实现:

    1. function inherit(Sub, Sup) {
    2. // Object.create和原型式继承一样,返回一个参数对象的副本
    3. // 寄生式继承+原型链继承
    4. const prototype = Object.create(Sup.prototype)
    5. Sub.prototype = prototype // 改变子类构造函数的prototype
    6. Sub.prototype.constructor = Sub
    7. Sub.__proto__ = Sup
    8. }
    9. function SupType(name) {
    10. this.name = name
    11. this.colors = ['red']
    12. }
    13. SupType.prototype.getName = function(){
    14. return this.name
    15. };
    16. function SubType(name,age) {
    17. this.age = age
    18. // 构造函数继承
    19. SupType.call(this, name)
    20. }
    21. inherit(SubType, SupType)
    22. SubType.prototype.getAge = function() {
    23. return this.age
    24. }
    25. const sub = new SubType('streetex', 25)