摘要:由于原型即本身也是對象,所以原型繼承可認為是一種特殊的對象式繼承。原型繼承里的原型即是函數(shù)的特有屬性,原型繼承事先得有函數(shù)。揭秘魔術箱不論原型繼承還是對象式繼承,其核心技術是實現(xiàn)了對象實例的鏈。
JavaScript的原型繼承是老生常談。由于原型即prototype本身也是對象,所以“原型”繼承可認為是一種特殊的“對象式”繼承?!睂ο笫健袄^承是筆者基于自己的理解,所提出的一個名詞。本文就著重闡述這兩種繼承方式的異同之處。
原型繼承JavaScript里的原型即prototype是函數(shù)的特有屬性,原型繼承事先得有函數(shù)。
// 定義函數(shù)Foo function Foo(name) { this.name = name; } // 定義Foo的原型 Foo.prototype.say = function() { console.log(this.name, "say"); }
函數(shù)及其原型定義好了,就可以使用原型繼承了。
var foo = new Foo("foo"); foo.say() //foo say foo instanceof Foo; //true。 foo 是Foo的一個實例 foo.__proto__ === Foo.prototype // true
上面用的是new操作符,它實際是通過Object.create工作,其過程如下。所以new其實是Object.create的便利操作方式。
var foo = Object.create(Foo.prototype); foo.name = "foo" foo.say(); foo instanceof Foo; //true。 foo 是Foo的一個實例 foo.__proto__ === Foo.prototype // true
請打起精神,Object.create(...)接受一個函數(shù)原型即object實例,以此實例為“模型”,創(chuàng)建新的object實例。新object實例的__proto__指向前function的prototype,自此新object實例也就擁有了前function prototype的字段和方法。
既然Foo.prototype本身是object實例,那么我們是否可以給Object.create(...)傳入一個普通的object實例呢?答案是可以的。這就是本文所要表述的“對象式”繼承。
“對象式“繼承開門見山地用code來說明:
// 創(chuàng)建對象實例Foo var Foo = { name: "foo", say: function() { console.log(this.name, "say"); } } // 以Foo為“模型”,創(chuàng)建新的對象實例foo var foo = Object.create(Foo); foo.__proto__ == Foo;// true // 所以foo也會擁有Foo的字段和方法,這點與原型繼承類似。 foo.say(); // foo say. // 但是foo不是Foo的實例。 foo instanceof Foo; // TypeError: Right-hand side of "instanceof" is not callable
如果Foo是個函數(shù),結果基本是相同的,除了instanceof Foo 會等于false。
function Foo() { } Foo.say = function() { console.log(Foo.name, "say"); } // 以Foo為“模型”,創(chuàng)建新的對象實例foo var foo = Object.create(Foo); foo.__proto__ == Foo;// true // 所以foo也會擁有Foo的字段和方法,這點與原型繼承類似。 foo.say(); // Foo say. // 但是foo不是Foo的實例。 foo instanceof Foo; // false“對象式”繼承的一個應用
舉個不太恰當?shù)睦樱绻覀冃枰獙ath.abs做些“修正”,對于在-1和1之間的數(shù)值保持原值。
var MMath = Object.create(Math); MMath.abs = function(val) { if (val >= -1 && val <=1) { return val; } return MMath.__proto__.abs(val); } MMath.abs(0.5) // 0.5 MMath.abs(-0.5) // -0.5 MMath.abs(-2) // 2揭秘Object.create魔術箱
不論原型繼承還是“對象式”繼承,其核心技術是Object.create(...)實現(xiàn)了對象實例的__proto__鏈。我們做個簡單的實現(xiàn):
Object.myCreate = function(obj) { if (obj instanceof Object || obj === null) { var newObj = {}; Object.setPrototypeOf(newObj, obj) return newObj; } else { throw "error happens. Should input a object"; } }
function Fooo() {} Fooo.say = function() { console.log(Fooo.name, "say"); } var myFooo = Object.myCreate(Fooo); myFooo.say(); // Fooo say
自此我們了解了Object.create(...)的黑魔法,也有助于我們理解Object.create({})和Oject.create(null)的區(qū)別,就是前者的__proto__是個object實例,擁有toString等方法。后者的__proto__是null,它不具有任何方法。在特別注重效率的情景,后者具有優(yōu)勢。
總結本文提出“對象式”繼承的概念,并與原型繼承對比,闡述其區(qū)別和聯(lián)系,希望有助于深化理解。雖然ES6越來越廣泛應用了,了解函數(shù)原型等這些ES3/ES5的概念應該是有助于JS的深入學習。
文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/84741.html
摘要:面向對象實現(xiàn)代碼動物發(fā)聲汪汪喵喵調用代碼動物發(fā)聲喵喵動物發(fā)聲汪汪當要增加一種動物時,只需增加一個繼承,不會影響其他已有的動物邏輯。所以的繼承和的原型繼承,可謂殊途同歸。 傳統(tǒng)面向對象的繼承和多態(tài) 我們知道C++/Java/C#等面向對象語言,都原生地支持類的繼承。繼承的核心作用大抵是創(chuàng)建一個派生類,并使其復用基本類(即父類)的字段和/或方法。并且派生類可以重寫基本類的方法。這樣基本類和...
摘要:從運行結果可以看出,當子類繼承多個父類的時候,對于構造函數(shù),只有第一個能夠被繼承,第二個就等掉了。重點看,類繼承了,同時,在構造函數(shù)中自己做了規(guī)定,也就是的構造函數(shù)是按照的意愿執(zhí)行,不執(zhí)行的內容,但是,還有一個方法,則繼承了這個方法。 在上一講代碼的基礎上,做進一步修改,成為了如下程序,請看官研習這個程序: #!/usr/bin/env python #coding:utf-8 c...
摘要:的內存分配方式修飾變量通常情況下,變量有個地方可以賦值直接賦值,構造函數(shù)中,或是初始化塊中。如就是對于變量,在聲明時,如果你沒有賦值,系統(tǒng)默認這是一個空白域,在構造函數(shù)進行初始化,如果是靜態(tài)的,則可以在初始化塊。 【java中為什么會有final變量】: final這個關鍵字的含義是這是無法改變的或者終態(tài)的; 那么為什么要阻止改變呢? java語言的發(fā)明者可能由于兩個目的而阻止改變: ...
摘要:在面向對象的程序設計語言中,多態(tài)是繼數(shù)據(jù)抽象和繼承之后的第三種基本特征。 在面向對象的程序設計語言中,多態(tài)是繼數(shù)據(jù)抽象和繼承之后的第三種基本特征。 1.再論向上轉型 多態(tài)作用:消除類型之間的耦合關系. 2.轉機 綁定:將一個方法調用同一個方法主體關聯(lián)起來. 前期綁定:在程序執(zhí)行前就進行綁定(面向過程語言默認綁定方式). 后期綁定:也叫動態(tài)綁定或運行時綁定,在運行時根據(jù)對象的類型進行綁...
摘要:這正是我們想要的太棒了毫不意外的,這種繼承的方式被稱為構造函數(shù)繼承,在中是一種關鍵的實現(xiàn)的繼承方法,相信你已經(jīng)很好的掌握了。 你應該知道,JavaScript是一門基于原型鏈的語言,而我們今天的主題 -- 繼承就和原型鏈這一概念息息相關。甚至可以說,所謂的原型鏈就是一條繼承鏈。有些困惑了嗎?接著看下去吧。 一、構造函數(shù),原型屬性與實例對象 要搞清楚如何在JavaScript中實現(xiàn)繼承,...
閱讀 931·2021-11-24 09:38
閱讀 947·2021-11-23 09:51
閱讀 2956·2021-11-16 11:44
閱讀 1785·2021-09-22 15:52
閱讀 1688·2021-09-10 11:20
閱讀 1412·2019-08-30 13:47
閱讀 1309·2019-08-29 12:36
閱讀 3341·2019-08-26 10:43