中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

ES6類和繼承的實現原理是什么

發布時間:2020-10-16 16:12:11 來源:億速云 閱讀:219 作者:小新 欄目:web開發

這篇文章給大家分享的是有關ES6類和繼承的實現原理是什么的內容。小編覺得挺實用的,因此分享給大家做個參考。一起跟隨小編過來看看吧。

1、es6 class 使用

javascript使用的是原型式繼承,我們可以通過原型的特性實現類的繼承,
es6為我們提供了像面向對象繼承一樣的語法糖。

class Parent {
  constructor(a){
    this.filed1 = a;
  }
  filed2 = 2;
  func1 = function(){}
}

class Child extends Parent {
    constructor(a,b) {
      super(a);
      this.filed3 = b;
    }
  
  filed4 = 1;
  func2 = function(){}
}

下面我們借助babel來探究es6類和繼承的實現原理。

1.類的實現

轉換前:

class Parent {
  constructor(a){
    this.filed1 = a;
  }
  filed2 = 2;
  func1 = function(){}
}

轉換后:

function _classCallCheck(instance, Constructor) {
  if (!(instance instanceof Constructor)) {
    throw new TypeError("Cannot call a class as a function");
  }
}

var Parent = function Parent(a) {
  _classCallCheck(this, Parent);

  this.filed2 = 2;

  this.func1 = function () { };

  this.filed1 = a;
};

可見class的底層依然是構造函數:

1.調用_classCallCheck方法判斷當前函數調用前是否有new關鍵字。

構造函數執行前有new關鍵字,會在構造函數內部創建一個空對象,將構造函數的proptype指向這個空對象的_proto_,并將this指向這個空對象。如上,_classCallCheck中:this instanceof Parent 返回true。

若構造函數前面沒有new則構造函數的proptype不會不出現在this的原型鏈上,返回false。

2.將class內部的變量和函數賦給this

3.執行constuctor內部的邏輯

4.return this (構造函數默認在最后我們做了)。

2.繼承實現

轉換前:

class Child extends Parent {
    constructor(a,b) {
      super(a);
      this.filed3 = b;
    }
  
  filed4 = 1;
  func2 = function(){}
}

轉換后:

我們先看Child內部的實現,再看內部調用的函數是怎么實現的:

var Child = function (_Parent) {
  _inherits(Child, _Parent);

  function Child(a, b) {
    _classCallCheck(this, Child);

    var _this = _possibleConstructorReturn(this, (Child.__proto__ || Object.getPrototypeOf(Child)).call(this, a));

    _this.filed4 = 1;

    _this.func2 = function () {};

    _this.filed3 = b;
    return _this;
  }

  return Child;
}(Parent);

1.調用_inherits函數繼承父類的proptype。

_inherits內部實現:

function _inherits(subClass, superClass) {
  if (typeof superClass !== "function" && superClass !== null) {
    throw new TypeError("Super expression must either be null or a function, not " + typeof superClass);
  }
  subClass.prototype = Object.create(superClass && superClass.prototype, {
    constructor: { value: subClass, enumerable: false, writable: true, configurable: true }
  });
  if (superClass)
    Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass;
}

(1) 校驗父構造函數。

(2) 典型的寄生繼承:用父類構造函數的proptype創建一個空對象,并將這個對象指向子類構造函數的proptype。

(3) 將父構造函數指向子構造函數的_proto_(這步是做什么的不太明確,感覺沒什么意義。)

2.用一個閉包保存父類引用,在閉包內部做子類構造邏輯。

3.new檢查。

4.用當前this調用父類構造函數。

var _this = _possibleConstructorReturn(this, (Child.__proto__ || Object.getPrototypeOf(Child)).call(this, a));

這里的Child.__proto__ || Object.getPrototypeOf(Child)實際上是父構造函數(_inherits最后的操作),然后通過call將其調用方改為當前this,并傳遞參數。(這里感覺可以直接用參數傳過來的Parent)

function _possibleConstructorReturn(self, call) {
  if (!self) {
    throw new ReferenceError("this hasn't been initialised - super() hasn't been called");
  }
  return call && (typeof call === "object" || typeof call === "function") ? call : self;
}

校驗this是否被初始化,super是否調用,并返回父類已經賦值完的this。

5.將行子類class內部的變量和函數賦給this。

6.執行子類constuctor內部的邏輯。

可見,es6實際上是為我們提供了一個“組合寄生繼承”的簡單寫法。

3. super

super代表父類構造函數。

super.fun1() 等同于 Parent.fun1()  或 Parent.prototype.fun1()。

super() 等同于Parent.prototype.construtor()

當我們沒有寫子類構造函數時:

var Child = function (_Parent) {
  _inherits(Child, _Parent);

  function Child() {
    _classCallCheck(this, Child);

    return _possibleConstructorReturn(this, (Child.__proto__ || Object.getPrototypeOf(Child)).apply(this, arguments));
  }

  return Child;
}(Parent);

可見默認的構造函數中會主動調用父類構造函數,并默認把當前constructor傳遞的參數傳給了父類。

所以當我們聲明了constructor后必須主動調用super(),否則無法調用父構造函數,無法完成繼承。

典型的例子就是Reatc的Component中,我們聲明constructor后必須調用super(props),因為父類要在構造函數中對props做一些初始化操作。

感謝各位的閱讀!關于ES6類和繼承的實現原理是什么就分享到這里了,希望以上內容可以對大家有一定的幫助,讓大家可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

曲阳县| 枞阳县| 清苑县| 襄汾县| 会宁县| 饶阳县| 通江县| 新巴尔虎右旗| 彰化县| 来宾市| 通州区| 阜康市| 白城市| 马边| 玉田县| 克东县| 施秉县| 小金县| 周至县| 莫力| 鄂温| 和顺县| 琼中| 得荣县| 肇庆市| 青州市| 崇仁县| 武乡县| 涞水县| 金阳县| 凤台县| 泾源县| 中方县| 太保市| 枞阳县| 昭苏县| 高邮市| 峨眉山市| 蚌埠市| 华蓥市| 凉山|