首頁 web前端 js教程 JavaScript繼承基礎講解(原型鏈、借用建構子、混合模式、原型式繼承、寄生式繼承、寄生組合式繼承)_javascript技巧

JavaScript繼承基礎講解(原型鏈、借用建構子、混合模式、原型式繼承、寄生式繼承、寄生組合式繼承)_javascript技巧

May 16, 2016 pm 04:39 PM
繼承

說好的講解JavaScript繼承,可是遲到現在講解。廢話不多說,直接進入正題。

  既然你想了解繼承,證明你對JavaScript物件導向已經有一定的了解,如還有什麼不理解的可以參考《物件導向JS基礎講解,工廠模式、建構函式模式、原型模式、混合模式、動態原型模式》,接下來講一般透過那些方法完成JavaScript的繼承。

  原型鏈

  JavaScript中實現繼承最簡單的方式就是使用原型鏈,將子類型的原型指向父類型的實例即可,即“子類型.prototype = new 父類型();”,實現方法如下:

// 为父类型创建构造函数
function SuperType() {
  this.name = ['wuyuchang', 'Jack', 'Tim'];
  this.property = true;
}

// 为父类型添加方法
SuperType.prototype.getSuerperValue = function() {
  return this.property;
}

// 为子类型创建构造函数
function SubType() {
  this.test = ['h1', 'h2', 'h3', 'h4'];
  this.subproperty = false;
}

// 实现继承的关键步骤,子类型的原型指向父类型的实例
SubType.prototype = new SuperType();

// 在此处给子类型添加方法,一定要在实现继承之后,否则会在将指针指向父类型的实例,则方法为空
SubType.prototype.getSubValue = function() {
  return this.subproperty;
}


/* 以下为测试代码示例 */
var instance1 = new SubType();
instance1.name.push('wyc');
instance1.test.push('h5');
alert(instance1.getSuerperValue());    // true
alert(instance1.getSubValue());      // false
alert(instance1.name);          // wuyuchang,Jack,Tim,wyc
alert(instance1.test);          // h1,h2,h3,h4,h5


var instance2 = new SubType();
alert(instance2.name);          // wuyuchang,Jack,Tim,wyc
alert(instance2.test);          // h1,h2,h3,h4
登入後複製

可以看到如上的程式碼就是透過原型鏈實現的一個簡單的繼承,但看到測試程式碼範例中還是存在一些問題。相信看了我的博文《物件導向JS基礎講解,工廠模式、建構函式模式、原型模式、混合模式、動態原型模式》的童鞋一定知道原型鏈程式碼存在的第一個問題是由於子類型的原型是父類型的實例,也就是子類型的原型中包含的父類型的屬性,從而導致引用類型值的原型屬性會被所有實例所共享。以上程式碼的instance1.name.push('wyc');就可以證明此問題的存在。而原型鏈的第二個問題就是:在建立子類型的實例時,不能傳遞參數給超類型的建構子。因此我們在實際開發中,很少單獨使用原型鏈。 

   借用建構子

  為了解決原型鏈中存在的兩個問題,開發人員開始使用一種稱為借用構造函數的技術來解決原型鏈中存在的問題。這種技術的實作想法也挺簡單,只需要在子類型的建構子內呼叫父類型的建構子。別忘了,函數只不過是在特定環境中執行程式碼的對象,因此可以透過apply()或call()方法執行建構子。程式碼如下:

// 为父类型创建构造函数
function SuperType(name) {
  this.name = name;
  this.color = ['pink', 'yellow'];
  this.property = true;

  this.testFun = function() {
    alert('http://tools.jb51.net/');
  }
}

// 为父类型添加方法
SuperType.prototype.getSuerperValue = function() {
  return this.property;
}

// 为子类型创建构造函数
function SubType(name) {
  SuperType.call(this, name);
  this.test = ['h1', 'h2', 'h3', 'h4'];
  this.subproperty = false;
}

// 在此处给子类型添加方法,一定要在实现继承之后,否则会在将指针指向父类型的实例,则方法为空
SubType.prototype.getSubValue = function() {
  return this.subproperty;
}


/* 以下为测试代码示例 */
var instance1 = new SubType(['wuyuchang', 'Jack', 'Nick']);
instance1.name.push('hello');
instance1.test.push('h5');
instance1.color.push('blue');
instance1.testFun();            // http://tools.jb51.net/
alert(instance1.name);            // wuyuchang,Jack,Nick,hello
// alert(instance1.getSuerperValue());    // error 报错
alert(instance1.test);            // h1,h2,h3,h4,h5    
alert(instance1.getSubValue());        // false    
alert(instance1.color);            // pink,yellow,blue

var instance2 = new SubType('wyc');
instance2.testFun();            // http://tools.jb51.net/
alert(instance2.name);            // wyc    
// alert(instance2.getSuerperValue());    // error 报错
alert(instance2.test);            // h1,h2,h3,h4
alert(instance2.getSubValue());        // false
alert(instance2.color);            // pink,yellow
登入後複製

可以看到以上程式碼中子型別SubType的建構函式內透過呼叫父型別"SuperType.call(this, name);",從而實現了屬性的繼承,也可以在子型別建立實例的時候為父類型傳遞參數了,但新的問題又來了。我可以看到我在父型別的建構子中定義了一個方法:testFun,在父型別的原型中定義了一個方法:getSuperValue。可是在實例化子型別後還是無法呼叫父型別的原型中定義的方法getSuperValue,只能呼叫父型別中建構函式的方法:testFun。這就同創建物件中只使用建構函式模式一樣,使得函數沒有復用性可言。考慮到這些問題,借用構造函數的技術也是很少單獨使用的。

組合繼承(原型鏈 借用建構子)

  顧名思義,組合繼承就是結合使用原型鏈與借用建構函式的優點,組合而成的一個模式。實作也很簡單,既然是結合,那當然結合了兩方的優點,即原型鏈繼承方法,而在構造函數繼承屬性。具體程式碼實作如下:

// 为父类型创建构造函数
function SuperType(name) {
  this.name = name;
  this.color = ['pink', 'yellow'];
  this.property = true;

  this.testFun = function() {
    alert('http://tools.jb51.net/');
  }
}

// 为父类型添加方法
SuperType.prototype.getSuerperValue = function() {
  return this.property;
}

// 为子类型创建构造函数
function SubType(name) {
  SuperType.call(this, name);
  this.test = ['h1', 'h2', 'h3', 'h4'];
  this.subproperty = false;
}

SubType.prototype = new SuperType();

// 在此处给子类型添加方法,一定要在实现继承之后,否则会在将指针指向父类型的实例,则方法为空
SubType.prototype.getSubValue = function() {
  return this.subproperty;
}


/* 以下为测试代码示例 */
var instance1 = new SubType(['wuyuchang', 'Jack', 'Nick']);
instance1.name.push('hello');
instance1.test.push('h5');
instance1.color.push('blue');
instance1.testFun();            // http://tools.jb51.net/
alert(instance1.name);            // wuyuchang,Jack,Nick,hello
alert(instance1.getSuerperValue());      // true
alert(instance1.test);            // h1,h2,h3,h4,h5    
alert(instance1.getSubValue());        // false    
alert(instance1.color);            // pink,yellow,blue

var instance2 = new SubType('wyc');
instance2.testFun();            // http://tools.jb51.net/
alert(instance2.name);            // wyc    
alert(instance2.getSuerperValue());      // true
alert(instance2.test);            // h1,h2,h3,h4
alert(instance2.getSubValue());        // false
alert(instance2.color);            // pink,yellow
登入後複製

以上程式碼透過SuperType.call(this, name);繼承父類型的屬性,透過SubType.prototype = new SuperType();繼承父類型的方法。以上程式碼很方便的解決了原型鏈與借用建構函數所遇到的問題,成為了JavaScript中最常用的實例繼承的方法。但混合模式也並非沒有缺點,可以看到在以上程式碼中在繼承方法的時候實際上已經繼承了父類型的屬性,只不過此時對於引用型別屬於​​共享的,因此在子類型的建構函式內在次調用父型別的建構子因而繼承了父型別的屬性而去覆寫了原型中所繼承的屬性,這樣呼叫兩次建構子顯然沒有必要,但有什麼方法可以解呢?解決此問題時先看以下兩個模式。

原型式繼承

  原型式繼承的的實現方法與普通繼承的實現方法不同,原型式繼承並沒有使用嚴格意義上的構造函數,而是藉助原型可以基於已有的對象創建新對象,同時還不必因此創建自訂類型。具體程式碼如下:

function object(o) {
  function F() {}
  F.prototype = o;
  return new F();
}
登入後複製

程式碼範例:

/* 原型式继承 */
function object(o) {
  function F() {}
  F.prototype = o;
  return new F();
}

var person = {
  name : 'wuyuchang',
  friends : ['wyc', 'Nicholas', 'Tim']
}

var anotherPerson = object(person);
anotherPerson.name = 'Greg';
anotherPerson.friends.push('Bob');

var anotherPerson2 = object(person);
anotherPerson2.name = 'Jack';
anotherPerson2.friends.push('Rose');

alert(person.friends);  // wyc,Nicholas,Tim,Bob,Rose

登入後複製

寄生式繼承

/* 寄生式继承 */
function createAnother(original) {
  var clone = object(original);
  clone.sayHi = function() {
    alert('hi');
  }
  return clone;
}
登入後複製

使用範例:

/* 原型式继承 */
function object(o) {
  function F() {}
  F.prototype = o;
  return new F();
}
   
/* 寄生式继承 */
function createAnother(original) {
  var clone = object(original);
  clone.sayHi = function() {
    alert('hi');
  }
  return clone;
}

var person = {
  name : 'wuyuchang',
  friends : ['wyc', 'Nicholas', 'Rose']
}
var anotherPerson = createAnother(person);
anotherPerson.sayHi();
登入後複製

寄生组合式继承

  前面说过了JavaScrip中组合模式实现继承的缺点,现在我们就来解决它的缺点,实现思路是,对于构造函数继承属性,而原型链的混成形式继承方法,即不用在继承方法的时候实例化父类型的构造函数。代码如下:

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

/* 寄生组合式继承 */
function inheritPrototype(subType, superType) {
  var prototype = object(superType.prototype);
  prototype.constructor = subType;
  subType.prototype = prototype;
}
登入後複製

而在使用时只需要将组合模式中的“SubType.prototype = new SuperType();”这行代码替换成inheritPrototype(subType, superType);即可。寄生组合式继承的高效率体现在它只调用了一次父类型构造函数,避免了创建不必要的或多余的属性。与此同时,原型链还能保持不变,因此,还能够正常使用instanceof和isPrototypeof()。这也是目前来说最理想的继承方式了,目前也在向这种模式转型。(YUI也使用了这种模式。)

此博文参考《JavaScript高级程序设计第3版》,代码为经过改写,更具体,并加了注释使大家更易懂。如对JS继承方面有独到见解的童鞋不别吝啬,回复您的见解供大家参考!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

C++ 函式繼承詳解:如何在繼承中使用「基底類別指標」和「衍生類別指標」? C++ 函式繼承詳解:如何在繼承中使用「基底類別指標」和「衍生類別指標」? May 01, 2024 pm 10:27 PM

在函數繼承中,使用「基底類別指標」和「衍生類別指標」來理解繼承機制:基底類別指標指向派生類別物件時,執行向上轉型,只存取基底類別成員。派生類別指標指向基底類別物件時,執行向下轉型(不安全),必須謹慎使用。

解決PHP報錯:繼承父類別時遇到的問題 解決PHP報錯:繼承父類別時遇到的問題 Aug 17, 2023 pm 01:33 PM

解決PHP報錯:繼承父類別時遇到的問題在PHP中,繼承是重要的物件導向程式設計的特性。透過繼承,我們能夠重複使用現有的程式碼,並且能夠在不修改原有程式碼的情況下,對其進行擴展和改進。儘管繼承在開發中應用廣泛,但有時在繼承父類別時可能會遇到一些報錯問題,本文將圍繞解決繼承父類別時遇到的常見問題進行討論,並提供相應的程式碼範例。問題一:未找到父類別在繼承父類別的過程中,如果系統無

C++ 函式繼承詳解:如何偵錯繼承中出現的錯誤? C++ 函式繼承詳解:如何偵錯繼承中出現的錯誤? May 02, 2024 am 09:54 AM

繼承錯誤調試技巧:確保正確的繼承關係。使用偵錯器逐步執行程式碼,檢查變數值。確保正確使用virtual修飾符。檢查隱藏的繼承帶來的菱形繼承問題。檢查抽象類別中未實現的純虛函數。

使用繼承的Java程式來計算定期存款(FDs)和定期存款(RDs)的利息 使用繼承的Java程式來計算定期存款(FDs)和定期存款(RDs)的利息 Aug 20, 2023 pm 10:49 PM

繼承是一個概念,它允許我們從一個類別存取另一個類別的屬性和行為。被繼承方法和成員變數的類別稱為超類別或父類,而繼承這些方法和成員變數的類別稱為子類別或子類別。在Java中,我們使用「extends」關鍵字來繼承一個類別。在本文中,我們將討論使用繼承來計算定期存款和定期存款的利息的Java程式。首先,在您的本機機器IDE中建立這四個Java檔案-Acnt.java−這個檔案將包含一個抽象類別‘Acnt’,用於儲存帳戶詳情,如利率和金額。它還將具有一個帶有參數‘amnt’的抽象方法‘calcIntrst’,用於計

C++ 函式繼承詳解:如何理解繼承中的「is-a」與「has-a」關係? C++ 函式繼承詳解:如何理解繼承中的「is-a」與「has-a」關係? May 02, 2024 am 08:18 AM

C++函式繼承詳解:掌握「is-a」和「has-a」關係什麼是函式繼承?函數繼承是C++中一種將衍生類別中定義的方法與基底類別中定義的方法關聯起來的技術。它允許衍生類別存取和重寫基底類別的方法,從而擴展了基底類別的功能。 「is-a」和「has-a」關係在函數繼承中,「is-a」關係指派生類別是基底類別的子類型,也就是說,衍生類別「繼承」了基底類別的特性和行為。 「has-a」關係指派生類別包含對基底類別物件的參考或指針,也就是說,衍生類別「擁有」了基底類別物件。語法以下是如何實作函數繼承的語法:classDerivedClass:pu

如何在PHP中使用多態性和繼承來處理資料類型 如何在PHP中使用多態性和繼承來處理資料類型 Jul 15, 2023 pm 07:41 PM

如何在PHP中使用多態性和繼承來處理資料類型引言:在PHP中,多型和繼承是兩個重要的物件導向程式設計(OOP)概念。透過使用多型和繼承,我們可以更靈活地處理不同的資料類型。本文將介紹如何在PHP中使用多態性和繼承來處理資料類型,並透過程式碼範例展示它們的實際應用。一、繼承的基本概念繼承是物件導向程式設計中的重要概念,它允許我們建立一個類,該類別可以繼承父類別的屬性和方法

C++ 中繼承和多態性如何影響類別的耦合度? C++ 中繼承和多態性如何影響類別的耦合度? Jun 05, 2024 pm 02:33 PM

繼承和多態性會影響類別的耦合度:繼承會增加耦合度,因為衍生類別依賴基底類別。多態性可以降低耦合度,因為物件可以透過虛擬函數和基底類別指標以一致的方式回應訊息。最佳實踐包括謹慎使用繼承、定義公共介面、避免在基底類別中新增資料成員,以及透過依賴注入解耦類別。實戰案例顯示如何使用多態性和依賴注入來降低銀行帳戶應用程式中的耦合度。

如何使用Java強制繼承代理final類別? 如何使用Java強制繼承代理final類別? Sep 06, 2023 pm 01:27 PM

如何使用Java強制繼承代理final類別?在Java中,final關鍵字用於修飾類別、方法和變量,表示它們不可被繼承、重寫和修改。然而,在某些情況下,我們可能需要強制繼承一個final類,以實現特定的需求。本文將討論如何使用代理模式來實現這樣的功能。代理模式是一種結構型設計模式,它允許我們建立一個中間物件(代理物件),該物件可以控制對另一個物件(被代理物件)的

See all articles