摘要:上圖中的在原型繼承稱作構造器。構造器就是一個普通的函數,但是將操作符用到構造器上時,它會執行一個叫的過程。從第條可以看到,構造器生成的對象的屬性會指向構造器的值,這就是我們構造原型鏈的關鍵。
基于類的繼承是大多數人所熟悉的,也是比較容易理解的。當我們形成類型繼承的思維定勢后,再次接觸原型繼承可能會覺得有些奇怪并難以理解。你更可能會吐槽,原型繼承根本就不能叫做繼承,一點都不面向對象。本人最初也是這樣認為的,但深入仔細的對比后發現,兩者其實并沒有本質的差別,只是表面有點不一樣而已。且看下面的分析。
類型繼承先看一個類型繼承的例子,代碼如下:
public class A { //... } public class B extends A { //... } public class C extends B { //... } C c = new C();
A、B、C為三個繼承關系的類,最后將類C實例化。下面這張圖描述了類和實例的對應關系。左邊為類,右邊為其對應實例。
我們看到,類C實例化后,內存中不僅存在c對象,同時還有a、b兩個對象。因為在java中,當我們在執行new C()操作時,jvm中會發生如下過程:
創建A的實例a。
創建B的實例b,并將實例b的super指針指向a。
創建C的實例c,并將實例c的super指針指向b。
過程1和過程2對用戶是透明的,不需要人工干預,引擎會按照“藍圖”把這兩個過程完成。通過上圖右半部分我們可以看到,super指針將a、b、c三個實例串起來了,這里是實現繼承的關鍵。當我們在使用實例c的某個屬性或方法時,若實例c中不存在則會沿著super指針向父類對象查找,直到找到,找不到則出錯。這就是繼承能夠達到復用目的內部機制。看到這里大家或許已經聯想到原型鏈了,super所串起來的這個鏈幾乎和原型鏈一樣,只是叫法不一樣而已。下面我們就來看看原型繼承。
原型繼承上面是原型繼承的示意圖。先看圖的右半部分,__proto__指針形成的對象鏈就是原型鏈。__proto__是一個私有屬性,只能看不準訪問(某些瀏覽器看也不給看)。__proto__的作用和前面的super是一樣的,原型鏈實現復用的機制和類型繼承也幾乎是一樣的,這里不再重復。有一點不一樣就是原型繼承中的屬性寫操作只會改變當前對象并不會影響原型鏈上的對象。
如何去構造原型鏈呢?看上去要稍微麻煩一些。原型繼承里面沒有類的概念,我們需要通過代碼,手動完成這個過程。上圖中的A、B、C在原型繼承稱作構造器。構造器就是一個普通的函數,但是將new操作符用到構造器上時,它會執行一個叫[[construct]]的過程。大致如下:
創建一個空對象obj。
設置obj的內部屬性[[Class]]為Object。
設置obj的內部屬性[[Extensible]]為true。
設置obj的[[__proto__]]屬性:如果函數對象prototype的值為對象則直接賦給obj,否則賦予Object的prototype值。
調用函數對象的[[Call]]方法并將結果賦給result。
如果result為對象則返回result,否則返回obj。
從第4條可以看到,構造器生成的對象的__proto__屬性會指向構造器的prototype值,這就是我們構造原型鏈的關鍵。下面的代碼是上圖原型鏈的構造過程。
function A(){ //... } function B(){ //... } function C(){ //... } var a = new A(); B.prototype = a; var b = new B(); C.prototype = b; var c = new C();
上述代碼雖然能達到目的,但有點繁瑣,我們可以將這個過程封裝一下。backbone的實現是這樣的:
var extend = function(protoProps, staticProps) { var parent = this; var child; if (protoProps && _.has(protoProps, "constructor")) { child = protoProps.constructor; } else { child = function(){ return parent.apply(this, arguments); }; } _.extend(child, parent, staticProps); child.prototype = _.create(parent.prototype, protoProps); child.prototype.constructor = child; child.__super__ = parent.prototype; return child; }
其中_.extend(child, parent, staticProps)是將staticProps和parent對象的屬性復制給child。_.create方法的實現大概如下。
_.create = function(prototype, protoProps){ var F = function(){}; F.prototype = prototype; var result = new F(); return _.extend(result, protoProps); }
有了extend方法,我們的代碼就可以寫成:
A.extend = extend; var B = A.extend({ //... ); var C = B.extend({ //... ); var c = new C();
這段代碼和類型繼承的代碼十分相似,通過原型繼承我們也可以達到類型繼承的效果。但是通過前面的比較我們發現,繼承的本質就其實就是對象的復用。原型繼承本身就是以對象為出發點考慮的,所以大多時候我們并不一定要按照類型繼承的思維考慮問題。而且js是弱類型,對象的操作也極其自由,上述的_.create方法可能是js里面實現繼承的一個更簡單有效的方法。
總結前面討論了兩種繼承方式,可以看到,繼承的本質其實就是對象的復用。本人覺得原型繼承更加的簡單和明確,它直接就是從對象的角度考慮問題。當然,如果你需要一個非常強大的繼承體系,你也可以構造出一個類似類型繼承的模式。相對來說,本人覺得原型繼承更靈活和自由些,也是非常巧妙和獨特的。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/64688.html
摘要:上圖中的在原型繼承稱作構造器。構造器就是一個普通的函數,但是將操作符用到構造器上時,它會執行一個叫的過程。從第條可以看到,構造器生成的對象的屬性會指向構造器的值,這就是我們構造原型鏈的關鍵。 基于類的繼承是大多數人所熟悉的,也是比較容易理解的。當我們形成類型繼承的思維定勢后,再次接觸原型繼承可能會覺得有些奇怪并難以理解。你更可能會吐槽,原型繼承根本就不能叫做繼承,一點都不面向對象。本人...
摘要:通常有這兩種繼承方式接口繼承和實現繼承。理解繼承的工作是通過調用函數實現的,所以是寄生,將繼承工作寄托給別人做,自己只是做增強工作。適用基于某個對象或某些信息來創建對象,而不考慮自定義類型和構造函數。 一、繼承的概念 繼承,是面向對象語言的一個重要概念。通常有這兩種繼承方式:接口繼承和實現繼承。接口繼承只繼承方法簽名,而實現繼承則繼承實際的方法。 《JS高程》里提到:由于函數沒有簽名,...
摘要:如下所示在規范中,已經正式把屬性添加到規范中也可以通過設置和獲取對象的原型對象對象之間的關系可以用下圖來表示但規范主要介紹了如何利用構造函數去構建原型關系。 前言 在軟件工程中,代碼重用的模式極為重要,因為他們可以顯著地減少軟件開發的成本。在那些主流的基于類的語言(比如Java,C++)中都是通過繼承(extend)來實現代碼復用,同時類繼承引入了一套類型規范。而JavaScript是...
摘要:深入之繼承的多種方式和優缺點深入系列第十五篇,講解各種繼承方式和優缺點。對于解釋型語言例如來說,通過詞法分析語法分析語法樹,就可以開始解釋執行了。 JavaScript深入之繼承的多種方式和優缺點 JavaScript深入系列第十五篇,講解JavaScript各種繼承方式和優缺點。 寫在前面 本文講解JavaScript各種繼承方式和優缺點。 但是注意: 這篇文章更像是筆記,哎,再讓我...
摘要:在規范中,引入了的概念。使用中的聲明一個類,是非常簡單的事。中面向對象實例化的背后原理,實際上就是原型對象。與區別理解上述原理后,還需要注意與屬性的區別。實際上,在中,類繼承的本質依舊是原型對象。 在 ES6 規范中,引入了 class 的概念。使得 JS 開發者終于告別了,直接使用原型對象模仿面向對象中的類和類繼承時代。 但是JS 中并沒有一個真正的 class 原始類型, clas...
閱讀 2661·2023-04-25 15:22
閱讀 2834·2021-10-11 10:58
閱讀 1054·2021-08-30 09:48
閱讀 1861·2019-08-30 15:56
閱讀 1736·2019-08-30 15:53
閱讀 1102·2019-08-29 11:16
閱讀 1056·2019-08-23 18:34
閱讀 1644·2019-08-23 18:12