成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

ES5/ES6 的繼承

libin19890520 / 2694人閱讀

摘要:原型鏈構造函數原型實例的關系每個構造函數都有一個原型對象,原型對象都包含一個指向構造函數的指針,實例有一個指向原型對象的指針構造函數原型對象構造函數構造函數操作符實例對象構造函數實例對象原型對象如果試

原型鏈

構造函數/原型/實例 的關系

每個構造函數都有一個原型對象,原型對象都包含一個指向構造函數的指針,實例有一個指向原型對象的指針

構造函數 --(prototype)--> 原型對象 --(constructor)--> 構造函數

構造函數 --(new 操作符)--> 實例對象 --(constructor)--> 構造函數

實例對象 --(__proto__)--> 原型對象

如果試圖使用某個對象(實例)的某個屬性或方法,會首先在對象內部尋找該屬性,如果找不到去該對象的原型(instance.__proto__)里面去找,如果還找不到就繼續(xù)沿著 __proto__ 這個鏈條往上找,直到找到 Object.prototype 為止

javascript 里面一切皆對象,所以都可以從這個鏈條去出發(fā)

JavaScript 的繼承不同于傳統面向對象是靠類實現繼承,而是通過原型鏈實現繼承

ES5 繼承

拷貝式繼承(通過深拷貝實現繼承)

原型式繼承

缺點:只能繼承原型方法

借用構造函數繼承

缺點:只能繼承實例屬性

組合式繼承

缺點:無論在什么情況下,都會調用兩次構造函數(創(chuàng)建父類實例的時候,在子類構造函數內不調用父類構造函數時)

組合寄生式繼承 (比較完美的繼承,但不能繼承父類靜態(tài)方法、靜態(tài)屬性)

function Parent() {}

function Child() {
    // 繼承父類實例屬性
    Parent.call(this) // 如果父類有參數寫在 this 后面
}

// 繼承父類原型方法
Child.prototype = Object.create(Parent.prototype)

// 修正子類原型的 constructor 指向
Child.prototype.constructor = Child 
兩個注意點:

Object.create(proto, [propertiesObject]) MDN

創(chuàng)建一個新對象,使用現有的對象來提供新創(chuàng)建的的對象的 __proto__

Object.create(null) // 創(chuàng)建一個沒有原型的空對象
第二個參數可添加屬性描述符

js高級程序設計用一下代碼代替的這個方法
function createObject(P) {
    var F = function() {}
    F.prototype = P.prototype
    return new F()
}

為什么要修正子類原型的 constructor 指向? 阮一峰

簡單總結一下:
任何一個 prototype 對象都有一個 constructor 屬性,指向它的構造函數
更重要的是,每一個實例也有一個 constructor 屬性,默認調用 prototype 的 constructor 屬性

如果沒有修正的那行代碼,結果如下
var c = new C()
c.constructor === Child // false
Child.prototype.constructor === Child // false

c.constructor === Parent // true
Child.prototype.constructor === Parent // true

這顯然會導致繼承鏈的混亂(c 明明是用構造函數Child生成的),因此我們必須手動糾正

ES6 繼承
ES6 的繼承本質上還是借助原型鏈實現繼承
// 借助 class extends 關鍵字
class Parent {
    static sayAge() {
        return "18"
    }
      constructor(name) {
        this.name = "name"
    }
}

class Child extends Parent {
    constructor(name, age) {
        /**
         * 如果寫 constructor 必須調用 super 方法,這是因為子類自己的 this 對象,必須先通過父類構造函數完成塑造
         * 不寫 super 就得不到 this對象,new 的時候就會報錯
         * Uncaught ReferenceError: Must call super constructor in derived class before accessing "this" or returning from derived constructor
         * 
         * ES5 實質上先創(chuàng)造子類的實例對象 this,然后再將父類的方法添加到 this 上面 (Parent.call(this))
         * ES6 實質上先將父類實例對象的屬性和方法,加到 this 上面(必須先調用 super 方法),然后用子類構造函數修改 this
        */
        super(name, age)
          this.age = age
    }
}
// 注意點:es6 的繼承可以繼承父類的靜態(tài)方法和靜態(tài)屬性,而ES5的繼承不行
// 經過 extends 之后,Child.__proto__ ==== Parent // true
// Parent.__proto__ 返回 f() { [native code] }
// Parent.__proto__.__proto__ === Object.prototype
Babel 轉碼后的 ES6 繼承代碼
// 為方便觀看,對代碼做了一些美化和省略處理
"use strict";

// 檢查子類是否調用了 super 方法
function _possibleConstructorReturn(self, call) {
  if (call && (_typeof(call) === "object" || typeof call === "function")) {
    return call;
  }
  return _assertThisInitialized(self);
}

function _assertThisInitialized(self) {
  if (self === void 0) {
    throw new ReferenceError(
      "this hasn"t been initialised - super() hasn"t been called"
    );
  }
  return self;
}

// 獲取子類的原型鏈指向的對象即父類
function _getPrototypeOf(o) {
  _getPrototypeOf = Object.setPrototypeOf
    ? Object.getPrototypeOf
    : function _getPrototypeOf(o) {
        return o.__proto__ || Object.getPrototypeOf(o);
      };
  return _getPrototypeOf(o);
}

// 核心繼承方法
function _inherits(subClass, superClass) {
  // ...
  // 同 es5 繼承的那一段
  subClass.prototype = Object.create(superClass.prototype, {
    constructor: { 
        value: subClass, // 修正 constructor 指向
        writable: true, 
        configurable: true 
    }
  });
  // 實現靜態(tài)屬性和方法的繼承 原理為:Child.__proto__ = Parent
  // 即子類(子類現在相當于實例)的在往上的 prototype = Parent,即子類可以使用父類的靜態(tài)屬性和方法
  if (superClass) _setPrototypeOf(subClass, superClass);
}

function _setPrototypeOf(o, p) {
  _setPrototypeOf =
    Object.setPrototypeOf ||
    function _setPrototypeOf(o, p) {
      o.__proto__ = p;
      return o;
    };
  return _setPrototypeOf(o, p);
}

// ... 省略類的創(chuàng)建及檢測等方法

var Parent =
  /*#__PURE__*/
  (function() {
    _createClass(Parent, null, [
      {
        key: "sayAge",
        value: function sayAge() {
          return "18";
        }
      }
    ]);

    function Parent(name) {
      _classCallCheck(this, Parent);

      this.name = "name";
    }

    return Parent;
  })();

var Child =
  /*#__PURE__*/
  (function(_Parent) {
    _inherits(Child, _Parent);

    function Child(name, age) {
      var _this;

      _classCallCheck(this, Child);
      
      // 下面一行代碼即 調用 super 的效果,如果不調用 super 子類將沒有 this
      _this = _possibleConstructorReturn(this, _getPrototypeOf(Child).call(this, name, age));

      /***
        * 如果注釋源碼中的 super 將編譯為如下代碼
        * _this.age = age;
        * return _possibleConstructorReturn(_this);
        * 因為沒有調用 super 子類還沒有 this,所以下一行直接報錯
        *
        * 如果源碼中不寫 _this.age = age
        * 將直接進入 _assertThisInitialized 方法,然后報錯,沒有調用 super 方法
      */

      _this.age = age;
      return _this;
    }

    return Child;
  })(Parent);

var c = new Child();

// 如果不用 babel轉碼,直接在瀏覽器里運行,不寫 super,結果如下和 babel 轉義后報錯信息不一樣
// 原生報錯信息: Uncaught ReferenceError: Must call super constructor in derived class before accessing "this" or returning from derived constructor

原生構造函數不可通過 extends 實現繼承 ES6阮一峰

因為子類拿不到 原生父類內部的對象,即是通過 call 也不行

new 運算符
上面說了繼承,那產生實例的操作符 new 是什么原理?
var obj = {}
obj.__proto__ = Child.prototype
F.call(obj)

// 1. 創(chuàng)建一個空對象
// 2. 將這個空對象的 __proto__ 屬性 指向了構造函數的 prototype 屬性 上 ==> 繼承原型屬性方法
// 3. 將構造函數的 this 指針替換成了 obj(實例),再調用 構造函數       ===> 繼承實例屬性方法
與原型鏈有關的幾個方法

hasOwnProperty : 該方法只會查找對象本身是否有某屬性,不會去原型鏈上尋找

A.isPropertyOf(instanceA) : 判斷 A 是不是 instanceA 的原型對象

instanceof : 判斷對象是不是某個構造函數的實例

__proto__ 只是瀏覽器廠商的私有實現,規(guī)范并不支持,規(guī)范支持Object.getPrototypeOf 和 Object.setPrototypeOf

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規(guī)行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://systransis.cn/yun/105935.html

相關文章

  • 為什么都說js 里面任何對象最終都繼承了Object對象

    摘要:今天閑來無事,看見幾行小字。又說所有對象,繼承終是。強行押韻一波這首詩的意思就是說的我今天沒有什么事情,然后無意中又在網上看到了任何對象都是從對象繼承而來的這句話。一時興起,便去驗證這句話。 今天閑來無事,看見幾行小字。又說所有對象,繼承終是Obj?!?強行押韻一波 這首詩的意思就是說的我今天沒有什么事情,然后無意中又在網上看到了任何對象都是從Object對象繼承而來的這句話。一時興...

    Gemini 評論0 收藏0
  • 前端基礎進階(十四):es6常用基礎合集

    摘要:在繼承的構造函數中,我們必須如上面的例子那么調用一次方法,它表示構造函數的繼承,與中利用繼承構造函數是一樣的功能。 showImg(https://segmentfault.com/img/remote/1460000009078532); 在實際開發(fā)中,ES6已經非常普及了。掌握ES6的知識變成了一種必須。盡管我們在使用時仍然需要經過babel編譯。 ES6徹底改變了前端的編碼風格,...

    Ryan_Li 評論0 收藏0
  • ES5/ES6繼承區(qū)別

    摘要:聲明會提升,但是不會被初始化賦值,所以優(yōu)先初始化賦值,則會進入暫時性死區(qū),類似,變量內部啟動嚴格模式的所有方法包括靜態(tài)方法和示例方法都沒有原型對象,所以也沒有,不能使用來調用必須使用來調用內部無法重寫類名 class聲明會提升,但是不會被初始化賦值,所以優(yōu)先初始化賦值,則會進入暫時性死區(qū),類似let,const變量 const bar = new Bar(); // ok funct...

    Jaden 評論0 收藏0
  • JS-繼承(es5,es6)

    摘要:組合式繼承是最常用的繼承模式,但組合繼承使用過程中會被調用兩次一次是創(chuàng)建子類型的時候,另一次是在子類型構造函數的內部。 首先需要了解原型鏈機制: 原型鏈作為實現繼承的主要方法,其基本思想就是利用原型讓一個引用類型繼承另 一個引用類型的屬性和方法. 構造函數、原型、實例之間的關系: 每個構造函數都有一個原型對象(prototype),原型對象都包含一個指向構造函數的指針(constr...

    AZmake 評論0 收藏0
  • ES6入門06】:對象擴展

    摘要:對象擴展簡潔表示法屬性表達式值用中括號包起來,就是個表達式跟的功能是一樣的數組也是引用類型,值雖然都是空,但指向不同的內存地址實現對象的拷貝淺拷貝只拷貝對象自身的屬性,如果對象有繼承,繼承的屬性不會被拷貝只拷貝可枚舉屬性,不可枚舉屬性不會被 對象擴展 簡潔表示法 { let o = 1,k = 2; let es5 = { o: o, k...

    zsirfs 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<