JS: 繼承

来源:https://www.cnblogs.com/guolao/archive/2019/03/02/10461687.html
-Advertisement-
Play Games

繼承 記錄一下 javascript 的各種繼承方式,個人用得比較多的還是原型鏈繼承和 ES6 的 extends。 原型鏈繼承 缺點: 在創建 Child 的實例時,無法向 Parents 傳參 父類裡面的引用類型被共用,個例修改導致所有實例都被修改 借用構造函數 為瞭解決上面的問題 ,經典繼承方 ...


繼承

記錄一下 javascript 的各種繼承方式,個人用得比較多的還是原型鏈繼承和 ES6 的 extends。

原型鏈繼承


// 原型模式
function Parents() {
  this.name = 'Parents';
}
Parents.prototype.getName = function() {
  return this.name;
}

// 原型鏈繼承
function Child() {}
Child.prototype = new Parents();
Child.prototype.constructor = Child;

let son = new Child();
son.getName(); // Parents;

缺點:

  • 在創建 Child 的實例時,無法向 Parents 傳參
  • 父類裡面的引用類型被共用,個例修改導致所有實例都被修改
function Parents() {
  this.name = ['dad', 'mom'];
}

function Child() {}
Child.prototype = new Parents();
Child.prototype.constructor = Child;

// 無法向 Parents 傳參
let son = new Child();
let daughter = new Child();
// 父類裡面的引用類型被共用
son.name == daughter.name; // true
// 個例修改導致所有實例都被修改
son.name[0] = 'father';
daughter.name[0]; // "father"

借用構造函數


為瞭解決上面的問題 ,經典繼承方式被設計出來:

// 構造函數模式
function Parents(childName) {
  this.childName = childName;
  this.name = ['dad', 'mom'];
  this.getName = function() {
    return this.name;
  }
}

// 借用構造函數繼承
function Child(name) {
  Parents.call(this, name);
}

// 可以向 Parents 傳參
let son = new Child('son');
son.childName; // "son"

// 父類裡面的引用類型不會被共用
let daughter = new Child('daughter');
son.name[0] = 'father';
daughter.name[0]; // "dad"

缺點:方法都在構造函數里定義,每次創建 Child 實例,都要重新創建一次方法,函數無法得到復用。

組合繼承


為了彌補上面的缺點,於是乎組合繼承誕生了。

// 組合模式
function Parents(childName) {
  this.childName = childName;
  this.name = ['dad', 'mom'];
}
Parents.prototype.getName = function() {
  return this.name;
}

// 組合繼承
function Child(name) {
  Parents.call(this, name);
}
Child.prototype = new Parents();
Child.prototype.constructor = Child;

let son = new Child('son');
son.name[0] = 'father';

let daughter = new Child('daughter');
daufhter.name[0]; // "dad"

因為原型鏈能保持不變,所以instanceOfisProtypeOf()也能識別基於組合繼承創建的對象。

原型式繼承


function obj(o) {
  function F() {}
  F.prototype = 0;
  return new F();
}

這種方式就是在 obj 函數內部,創建一個臨時性的構造函數 F(),然後將傳入的對象作為這個構造函數的原型,最後返回這個臨時類的實例。

其實就是對傳入的對象進行一次淺複製,類似於 ES5 的 Object.create() 方法。

Object.create(proto, [propertiesObject]) 方法創建一個新對象,使用現有的對象來提供新創建的對象的__proto__

proto: 新創建對象的原型對象。

propertiesObject: 可選。如果沒有指定為 undefined,則是要添加到新創建對象的可枚舉屬性(即其自身定義的屬性,而不 是其原型鏈上的枚舉屬性)對象的屬性描述符以及相應的屬性名稱。這些屬性對應Object.defineProperties() 的第二個參數。

寄生式繼承


function createObj (o) {
    var clone = Object.create(o);
    clone.sayHi = function () {
        console.log('hi');
    }
    return clone;
}

這種方式就是創建一個僅用於封裝繼承過程的函數,該函數在內部以某種方式來增強對象,最後再返回增強過的對象。本質上還是對傳入的對象進行一次淺複製。

寄生組合式繼承


這種方式得先創建一個用於封裝繼承過程的函數。

function inheritPrototype(child, Parents) {
  let prototype = Object.create(Parents.prototype);
  prototype.constructor = child;
  child.prototype = prototype
}

然後用上面的函數實現繼承:

function Parents(childName) {
  this.childName = childName;
  this.name = ['dad', 'mom'];
}
Parents.prototype.getName = function() {
  return this.name;
}

function Child(name) {
  Parents.call(this, name);
}

inheritPrototype(Child, Parents);

其實和組合繼承差不多,只是封裝了一個 inheritPrototype 函數使得在繼承的時候只在 Child 里調用一次 Parents 構造函數。

ES6 的 extends


ES6 引入了 class,可以通過extends關鍵字實現繼承,這比 ES5 的通過修改原型鏈實現繼承,要清晰和方便很多。但需要註意的是,ES6 的 class 只是語法糖,本質上還是 ES5 的通過修改原型鏈實現繼承的方法,具體可以去babel 試用來看看 babel 對於 class 的轉換。

class Parents {
  // 構造函數
  constructor(name) {
    this.name = name;
  }
  // 靜態方法
  static getName() {
    return this.name;
  }
  // 方法
  printName() {
    console.log(this.name);
  }
}

class Child extends Parents {
  constructor(name, childName) {
    // 調用父類的constructor(name)
    super(name);
    this.childName = childName;
  }
  printName() {
    super.printName();
    console.log(this.childName);
  }
}

let c = new Child('mom', 'son');
c.printName();// mom son

ES5 的繼承,實質是先創造子類的實例對象this,然後再將父類的方法添加到this上面(Parent.apply(this))。ES6 的繼承機制完全不同,實質是先將父類實例對象的屬性和方法,加到this上面(所以必須先調用super方法),然後再用子類的構造函數修改this

大多數瀏覽器的 ES5 實現之中,每一個對象都有__proto__屬性,指向對應的構造函數的prototype屬性。Class 作為構造函數的語法糖,同時有prototype屬性和__proto__屬性,因此同時存在兩條繼承鏈。

(1)子類的__proto__屬性,表示構造函數的繼承,總是指向父類。

(2)子類prototype屬性的__proto__屬性,表示方法的繼承,總是指向父類的prototype屬性。

參考


阮一峰_Class 的繼承

《javascript高級程式設計》(三)


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • docker啟動mysql docker run -p 3306:3306 -v /dockermysqlcfg/config/my.cnf:/etc/mysql/my.cnf -v /dockermysqlcfg/data:/var/lib/mysql -e MYSQL_ROOT_PASSWORD ...
  • 在Eclipse中使用springboot整合Mybatis,連接到5.7版本Mysql報錯WARN: Establishing SSL connection without server's identity verification is not recommended. According t ...
  • https://blog.csdn.net/demonson/article/details/79474133 ...
  • RMAN優缺點及RMAN備份及恢復步驟--以下部分來自網路轉摘,僅供參考和OS命令備份方式相比,使用RMAN的優點1 備份執行期間不需要人工干預,因此減少了誤操作的機會;2 可以有效的將備份和恢復結合起來;3 支持除邏輯備份以外的所有備份類型,包括完全備份,增量備份,表空間備份,數據文件備份,控制文 ...
  • Android遠程桌面助手(B1413),優化性能,提高相容性和連接的可靠性,解決Android 9上顯示黑屏問題。 ...
  • 基本思路 聖杯佈局分為3段:上、中、下。 中段被分為:左、中、右3塊。 1:採用flex佈局時,先把彈性容器主軸設置為垂直方向(flex-direction:column) 2:上、中、下3塊彈性項目設置均勻拉伸(flex:1)或固定上、下兩端大小,讓中間自動拉伸。註意:flex:拉伸是方向為主軸方 ...
  • 1:創建一個彈性容器(display:flex) 2:構建2個或3個彈性項目. 3:把彈性項目設置為居中對齊.(align-items:center) 4:改變input自身對齊方式,把它設置為拉伸以適應容器(align-self:stretch)。 實例: ...
  • 解決問題:大同大學教務處官網教師埠一進去就卡住了,點上面一行的菜單無響應 教師端下載方法: 鏈接:https://pan.baidu.com/s/10E-_0C7xVcKvVIyWxWHEVw 提取碼:e1ik 學生端下載方法: 鏈接:https://pan.baidu.com/s/1MMmAVd ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...