首頁 web前端 js教程 javascript繼承為什麼要繼承_基礎知識

javascript繼承為什麼要繼承_基礎知識

May 16, 2016 pm 05:48 PM
繼承

Quiz1
Javascript真的需要類別(Class)麼?
我們先看下其他有類別(Class)的物件導向語言(如:Java)的一些特性。

父類別與子類別
父類別(Superclass)和子類別(Subclass),並不是為了解決父親與兒子的問題,而是為了解決類別的包含關係的,我們用Sub表示“子類”,用Sup表示“父類”,則有:
  Sub Sup
這是有區別的,例如通常我們能夠將子類當成父類來使用,但認人的時候我們並不能把兒子當成父親。
或者可以這麼說,父類別和子類別不是為了解決類別間存在相同方法或屬性的。

舉個例子
有人喜歡這樣做:
我們需要一些動物的類,以便在屏幕上創建一些移動的動物,但移動的動物有些在空中飛行,有些在路上行走。
所以建立兩個父類,一個是Fly,一個是Walk:
複製程式碼 程式碼如下:

Class Fly{
Fly(){}
}
Class Walk{
Walk(){}
}

然後獅子
}
然後獅子 } 然後獅子
}


然後獅子們(還可以再建些其他的在路上行走的動物)就屬於Walk類,老鷹們(也還可以再建些其他在天上飛行的動物)就屬於Fly類:



複製程式碼


程式碼如下:
Class Lion extend Walk{
}
Class Eagle extend Fly{
}
}
Class Eagle extend Fly{ >

最後對Lion和Eagle類別建立一些實例,呼叫對應的方法,螢幕上就會有一些獅子和老鷹在移動了。
但這可能不是一個好的設計,比如明天老闆突然一拍大腦,他要有一種叫天馬(Pegasus)的動物,它們即會在天上飛,又會在路上走,時而要飛行,時候要行走。
在這種情況下,這個方案就全然無用了。
為什麼這個設計失敗了?
繼承是有條件的,子類別必須能嚴格的向上轉換(變成父類別)。 在上面這個例子中: 獅子(Lion)被假設等同於行走動物(Walk),老鷹(Eagle)被假設等同於飛行動物(Fly)。 這看起來很成功,因為子類能嚴格向上轉型,但他有隱患。 當有一種天馬(Pegasus)介入到裡面的時候,我們才發現獅子其實只是“會行走的動物”,老鷹其實只是“會飛行的動物”,這不意味著動物一輩子只能飛行或行走,所以即會飛行又會行走的天馬就找不到自己的歸屬了。 這個例子很好的證明了,子類和父類不是為了解決類間具有相同的方法的: 一些動物都會行走,需要擁有行走(Walk)這個方法,但這不應該由子類別和父類別實作。

組合

我們可以這樣解決這個問題:




複製程式碼



複製程式碼


代碼如下:


Class Lion{
walker = new Walk();
walk(){
return walker.walk();
}
}
Class Eagle{ flyer = new Fly(); fly(){ return flyer.fly(); } } Class Pegasus{ walker = new Walk(); flyer = new Fly(); walk(){ return walker.walk(); } fly(){ return flyer. fly(); } }
組合是簡單的在新類別內部建立原有類別物件。所以組合才是為了解決類別間具有相同的方法的。在這個例子裡面:
Walk被當成“會行走的動物應該擁有的方法集合”,同理Fly被當成“會行走的動物應該擁有的方法集合”,所以對於天馬(Pegasus),我們只需要對Walk和Fly進行組合就行了。

繼承的目的
繼承並非程式碼多用的唯一方法,但繼承有他的優勢:
子類別可以向上轉換變成父類別。
這樣我們就可以忽略所有的子類別差異,當成相同的類別來操作,例如:
我們有方法fn(A),fn(B),這兩個方法實際上是相似的,我們想復用他們。
則我們可以透過設立一個父類C,其中A是C的子類,B是C的子類,那麼fn(C)就可以重複使用在A和B身上了。

回到Javascript
但回到Javascript,我們發現上面的範例是不成立的。
因為Javascript本身就是一種弱型別語言,它並不會在操作前(因為他不用編譯)專注在自己操作的物件類型是什麼。他只會執行成功,或發生錯誤。
這時候,繼承顯得並不必要了。那麼類別也就同樣不是必要的了。
I have been writing JavaScript for 8 years now, and I have never once found need to use an uber function. The super idea is fairly important in the classical pattern, but it appears to be unnecessary in the protox I now see my early attempts to support the classical model in JavaScript as a mistake.
——Douglas Crockford
我寫Javascript程式碼已經8年了,但我從來沒有發現需要使用超類別函數。超類的想法在古典設計模式是非常重要的,但這在以原型和函數為基調的模式中並不必要。我現在覺得,早期我試著讓Javascript支援經典模式是個錯誤的決定。

安全環境
當然,你可以手動去判斷類型,控制參數的類型,進而提供一個較為安全的環境。
例如同樣作為弱型別腳本語言的PHP,為了模擬強型態物件導向語言設定安全環境,只好這麼做:
複製程式碼 程式碼如下:

class ShopProductWriter{
public function write( $shopProduct ){
if( ! ( $Product nce 廣告購買 instact duct) &Product ) ){
die( "輸入錯誤的類型" );
}
//如果類型正確就執行一些代碼
}
}

— —PHP Objects, Patterns, and Practtice Third Edition . Matt Zandstra
但這只是一個非常醜陋的方案而已。

經典繼承語法糖實現
不過經典繼承依然是許多人喜歡的方式。所以YUI、Prototype、Dojo、MooTools都提供了自己的實作方案。
其中較為常見的方案中,語法大概是這樣的:
複製代碼 代碼如下:

var Person = Class.extend({
init: function(isDancing){
this.dancing = isDancing;
}
});
var Dancer = Person.extend ({
init: function(){
this._super( true );
}
});
var n = new Dancer();
alert(n.dancing ); //true

最重要的實現是對this._super的實現,其實extend函數只是將傳進來的對象重新組裝了一下,變成一個原型對象,新構造函數的prototype裡。
具體實作請查看參考文獻1。

ECMAScript 6的經典繼承語法糖
對於類別庫各自實現,導致經典繼承語法眾多,ECMA組織貌似不太滿意,他們試圖在ECMAScript 6中加入更加直觀的經典繼承語法糖:
複製程式碼 程式碼如下:

class Amal Amal {


class Amal Amal { constructor(name) {
this.name = name;
}
sayName() {
console.log(this.name);
}
}
class Dog extends Animal {
constructor(name) {
super(name);
}
bark() {
console.log("Woof!");
}
console.log("Woof!"); } }
總結
實際上,Javascript中經典繼承並不是必要的。
然而基於許多人喜歡經典繼承模型,所以在新版的ECMAScript 6中提供了相關語法糖。
不過在中國,前端想廣泛使用該語法糖應該是一個遙遠的故事…

Quiz2
那Javascript特有的繼承呢?

原型繼承
原型繼承不是解決經典繼承中的集合包含關係的,實際上原型繼承是解決從屬關係的,用數學表達就是:
  Sub. prototype ∈ Sup
子級建構函式(子型別)原型是父級建構子(父型別)建構的實例物件。原型實際上是子類型實例中需要共享的東西:
複製代碼 代碼如下:

function Being(){
this.living = true;
}
Being.prototype.walk = function(){
alert("I' m walking");
};
function Dancer(){
this.dancing = true;
}
Dancer.prototype = new Being();
Dancer.prototype.dance = function(){
alert ("I'm dancing");
};
var one = new Dancer();
one.walk();
one.dance();

利用借用、寄生等技術可以產生許多不同的繼承效果,但這些技術都只是為了解決原型繼承中屬性和方法一些公用與非公用的問題罷了。由於篇幅問題就不展開討論了,有興趣可以參考《Javascript高級程式設計》相關內容。

思考題
1.文章開頭關於天馬(Pegasus)的題目,如果在Javascript上,應該如何設計呢?例如我們有下列兩種:
複製程式碼 程式碼如下:

function Walk() {
this.walk = function(){
//walk
};
}
function Fly(){
this.fly = function(){
/ /fly
};
}
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡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