摘要:實(shí)例可以通過(guò)代理來(lái)找到它,并用來(lái)檢測(cè)其構(gòu)造函數(shù)。經(jīng)典繼承圖這也是通過(guò)構(gòu)造函數(shù)來(lái)創(chuàng)建對(duì)象,但是在這一系列的對(duì)象和實(shí)例之間我們的焦點(diǎn)是放在原型鏈上。盡管,但構(gòu)造函數(shù)的屬性并不是對(duì)象自己的屬性,它實(shí)際上是通過(guò)尋找原型鏈獲得的,即所指向的地方。
繼承是面向?qū)ο缶幊陶Z(yǔ)言的一大核心功能點(diǎn),雖然JavaScript并不是一門真正意義上的面向?qū)ο蟮木幊陶Z(yǔ)言,但也通過(guò)某種手段實(shí)現(xiàn)了繼承這一功能,最常見的JavaScript繼承是通過(guò)原型鏈去實(shí)現(xiàn),這就涉及到了prototype、__proto__、[[prototype]]三者之間的關(guān)系了。
如上圖所示,理解JavaScript中的繼承的關(guān)鍵是要理解母雞如何產(chǎn)蛋的過(guò)程。
每個(gè)對(duì)象都可以有另一個(gè)對(duì)象作為其原型。然后前一個(gè)對(duì)象繼承了它的所有原型屬性。對(duì)象通過(guò)內(nèi)部屬性[[Prototype]]指定其原型。由[[Prototype]]屬性連接的對(duì)象鏈稱為原型鏈。
為了理解基于原型的繼承是怎么樣的,我們先來(lái)看一個(gè)例子。
var proto = { describe: function () { return "name: "+this.name; } }; var obj = { [[Prototype]]: proto, name: "obj" }; > obj.describe [Function] > obj.describe() "name: obj"
__proto__是Object.prototype對(duì)象的訪問(wèn)者屬性。它暴露了訪問(wèn)它的對(duì)象的內(nèi)部原型([[Prototype]])
function Foo(name) { this.name = name this.whoAmI = function () { return this.name } } var b = new Foo("b") var a = new Foo("a") b.say = function () { console.log(`Hi from ${this.whoAmI()}`) } console.log(a.__proto__ === Foo.prototype); // true console.log(a.__proto__ === b.__proto__); // true
JavaScript引擎在對(duì)象b上添加了一個(gè)say方法,而不是在Foo原型對(duì)象(Foo.prototype)上。
正如上圖中看到的那樣,a.__ proto__暴露了指向Foo.prototype對(duì)象的[[Prototype]]。同樣,b.__ proto__也指向與a.__ proto__相同的對(duì)象。
通過(guò)構(gòu)造函數(shù)來(lái)創(chuàng)建對(duì)象除了通過(guò)指定模式創(chuàng)建對(duì)象之外,構(gòu)造函數(shù)還可以執(zhí)行另一個(gè)有用的操作 - 它會(huì)自動(dòng)為新創(chuàng)建的對(duì)象設(shè)置原型對(duì)象。此原型對(duì)象存儲(chǔ)在構(gòu)造函數(shù)的原型對(duì)象屬性中。
我們可以使用構(gòu)造函數(shù)用b對(duì)象重寫前面的例子。因此,對(duì)象a(原型)Foo.prototype的作用:
創(chuàng)建擁有原型對(duì)象屬性x及方法calculate()的一個(gè)Foo對(duì)象。
function Foo(y) { this.y = y; } Foo.prototype.x = 10; Foo.prototype.calculate = function (z) { return this.x + this.y + z; };
使用對(duì)象Foo創(chuàng)建一個(gè)b對(duì)象實(shí)例。
var b = new Foo(20); b.calculate(30); // 60 console.log( b.__proto__ === Foo.prototype, // true b.__proto__.calculate === Foo.prototype.calculate // true b.__proto__.calculate === b.calculate, // true Foo === b.constructor, // true Foo === Foo.prototype.constructor, // true );
正如上面打印出來(lái)的信息,對(duì)象b從Foo()繼承了方法calculate?!癋oo.prototype”自動(dòng)創(chuàng)建一個(gè)特殊屬性“constructor”,它是對(duì)構(gòu)造函數(shù)本身的引用。實(shí)例b可以通過(guò)代理來(lái)找到它,并用來(lái)檢測(cè)其構(gòu)造函數(shù)。
JavaScript經(jīng)典繼承圖這也是通過(guò)構(gòu)造函數(shù)來(lái)創(chuàng)建對(duì)象,但是在這一系列的對(duì)象和實(shí)例之間我們的焦點(diǎn)是放在原型鏈上。原型對(duì)象其實(shí)也是普通的對(duì)象,也有屬于它們自己的屬性。如果原型具有對(duì)其原型的非空引用,依此類推,則稱為原型鏈。
以下是JavaScript經(jīng)典繼承的圖表。構(gòu)造函數(shù)Foo只是虛構(gòu)類的類名。 foo對(duì)象是Foo的一個(gè)實(shí)例。
現(xiàn)在我們可以從圖中看到為什么當(dāng)我們從Dog類繼承Animal 時(shí),我們會(huì)這樣做:
function Dog() {} // the usual constructor function Dog.prototype = new Animal(); Dog.prototype.constructor = Dog;
當(dāng)通過(guò)new操作符創(chuàng)建一個(gè)實(shí)例的時(shí)候,都發(fā)生了些什么:
注意Foo.prototype的原型(prototype)并不是來(lái)自原型鏈。Foo.prototype指向原型鏈中的某些位置,但Foo的這個(gè)原型屬性不是來(lái)自原型鏈。構(gòu)成原型鏈的是proto指向鏈,以及proto指向的對(duì)象。比如從foo .__ proto__,鏈接到到foo .__ proto . proto__,等等,直到達(dá)到null(也就是咱們常說(shuō)的原型鏈的頂部是null)。
proto and prototype的關(guān)系:
JavaScript的經(jīng)典繼承就像這樣:我是一個(gè)構(gòu)造函數(shù),我只是一個(gè)函數(shù),我持有一個(gè)原型引用,每當(dāng)調(diào)用foo = new Foo()時(shí),我會(huì)讓foo .__ proto__指向我的原型對(duì)象。所以,F(xiàn)oo.prototype 與 obj.__proto__是兩個(gè)不同的概念。Foo.prototype指示,當(dāng)創(chuàng)建Foo的對(duì)象時(shí),這是新對(duì)象的原型鏈應(yīng)該指向的點(diǎn) - 也就是說(shuō),foo .__ proto__應(yīng)該指向Foo.prototype指向的位置。
如果需要添加一個(gè)方法怎么辦?
如果woofie這個(gè)對(duì)象沒(méi)有movie方法,它將通過(guò)原型鏈去尋找,就像任何原型繼承場(chǎng)景一樣,首先這個(gè)對(duì)象通過(guò)woofie.__proto__,它與Dog.prototype所指的對(duì)象相同。如果move方法不是這個(gè)對(duì)象的一個(gè)屬性(意味著Dog這個(gè)類并沒(méi)有move這個(gè)方法),則在原型鏈中上升一級(jí)(去原型鏈上尋找),即woofie .__ proto . proto__,或者與Animal.prototype相同。
Animal.prototype.move = function() { ... };
*盡管foo.constructor === Foo,但構(gòu)造函數(shù)的屬性并不是foo對(duì)象自己的屬性,它實(shí)際上是通過(guò)尋找原型鏈獲得的,即foo .__ proto__所指向的地方。對(duì)Function.constructor也是一樣。
當(dāng)我們看到Constructor.prototype,foo .__ proto__,F(xiàn)oo.prototype.constructor時(shí),該圖可能很復(fù)雜,有時(shí)會(huì)令人困惑。
要驗(yàn)證圖表,請(qǐng)注意,即使foo.constructor將顯示一個(gè)值,foo.constructor也不是foo自己的屬性,而是通過(guò)跟蹤原型鏈獲得的,因?yàn)榭梢詼贤ㄟ^(guò)foo.hasOwnProperty(“constructor”)來(lái)進(jìn)行驗(yàn)證。
*
[[Prototype]]:對(duì)象通過(guò)其內(nèi)部屬性指定的原型對(duì)象
proto :創(chuàng)建的對(duì)象實(shí)例所擁有的內(nèi)部屬性,在語(yǔ)言層面可以直接訪問(wèn)[[Prototype]]
prototype:prototype是在使用new操作符創(chuàng)建對(duì)象時(shí)用于構(gòu)建__proto__的對(duì)象,在實(shí)例化的對(duì)象上(或其他對(duì)象)不可使用,僅在構(gòu)造函數(shù)上使用,因?yàn)樗菑腇untion和Object上復(fù)制的。而__proto__隨處可用
( new Foo ).__proto__ === Foo.prototype //true ( new Foo ).prototype === undefined //true參考資料
http://speakingjs.com/es5/ind...
http://www.javascripttutorial...
http://stackoverflow.com/ques...
http://dmitrysoshnikov.com/ec...
https://www.quora.com/What-is...
http://www.jisaacks.com/proto...
http://kenneth-kin-lum.blogsp...
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/105828.html
摘要:在規(guī)范中,引入了的概念。使用中的聲明一個(gè)類,是非常簡(jiǎn)單的事。中面向?qū)ο髮?shí)例化的背后原理,實(shí)際上就是原型對(duì)象。與區(qū)別理解上述原理后,還需要注意與屬性的區(qū)別。實(shí)際上,在中,類繼承的本質(zhì)依舊是原型對(duì)象。 在 ES6 規(guī)范中,引入了 class 的概念。使得 JS 開發(fā)者終于告別了,直接使用原型對(duì)象模仿面向?qū)ο笾械念惡皖惱^承時(shí)代。 但是JS 中并沒(méi)有一個(gè)真正的 class 原始類型, clas...
摘要:可能有信息敏感的同學(xué)已經(jīng)了解到庫(kù)爆出嚴(yán)重安全漏洞,波及萬(wàn)項(xiàng)目。以此為例,可見這次漏洞算是比較嚴(yán)重了。此外,凍結(jié)一個(gè)對(duì)象后該對(duì)象的原型也不能被修改。 摘要: 詳解原型污染。 原文:Lodash 嚴(yán)重安全漏洞背后 你不得不知道的 JavaScript 知識(shí) 作者:Lucas HC Fundebug經(jīng)授權(quán)轉(zhuǎn)載,版權(quán)歸原作者所有。 可能有信息敏感的同學(xué)已經(jīng)了解到:Lodash 庫(kù)爆出嚴(yán)...
摘要:可能有信息敏感的同學(xué)已經(jīng)了解到庫(kù)爆出嚴(yán)重安全漏洞,波及萬(wàn)項(xiàng)目。以此為例,可見這次漏洞算是比較嚴(yán)重了。此外,凍結(jié)一個(gè)對(duì)象后該對(duì)象的原型也不能被修改。使用數(shù)據(jù)結(jié)構(gòu),不會(huì)存在原型污染狀況。 可能有信息敏感的同學(xué)已經(jīng)了解到:Lodash 庫(kù)爆出嚴(yán)重安全漏洞,波及 400萬(wàn)+ 項(xiàng)目。這個(gè)漏洞使得 lodash 連夜發(fā)版以解決潛在問(wèn)題,并強(qiáng)烈建議開發(fā)者升級(jí)版本。 我們?cè)诿χ礋狒[或者升級(jí)版本的同時(shí)...
摘要:目錄無(wú)繼承簡(jiǎn)單的字段聲明無(wú)繼承簡(jiǎn)單的方法聲明簡(jiǎn)單繼承一層繼承字段覆蓋無(wú)繼承靜態(tài)函數(shù)無(wú)繼承靜態(tài)變量神秘的類無(wú)繼承簡(jiǎn)單的字段聲明先來(lái)看個(gè)最簡(jiǎn)單的例子,我們僅僅使用了關(guān)鍵字并定義了一個(gè)變量最后編譯出來(lái)的代碼如下。無(wú)繼承靜態(tài)變量還有個(gè)小例子。 在[上一篇文章][]中,我們提到 ES6 的 class 語(yǔ)法糖是個(gè)近乎完美的方案,并且講解了實(shí)現(xiàn)繼承的許多內(nèi)部機(jī)制,如 prototype/__pro...
摘要:?jiǎn)栴}修改實(shí)例的,即修改了構(gòu)造函數(shù)的原型對(duì)象的共享屬性到此處,涉及到的內(nèi)容大家可以再回頭捋一遍,理解了就會(huì)覺得醍醐灌頂。 開場(chǎng)白 大三下學(xué)期結(jié)束時(shí)候,一個(gè)人跑到帝都來(lái)參加各廠的面試,免不了的面試過(guò)程中經(jīng)常被問(wèn)到的問(wèn)題就是JS中如何實(shí)現(xiàn)繼承,當(dāng)時(shí)的自己也是背熟了實(shí)現(xiàn)繼承的各種方法,回過(guò)頭來(lái)想想?yún)s不知道__proto__是什么,prototype是什么,以及各種繼承方法的優(yōu)點(diǎn)和缺點(diǎn),想必有好...
閱讀 3212·2021-11-08 13:18
閱讀 1365·2021-10-09 09:57
閱讀 1197·2021-09-22 15:33
閱讀 3996·2021-08-17 10:12
閱讀 5079·2021-08-16 11:02
閱讀 2693·2019-08-30 10:56
閱讀 975·2019-08-29 18:31
閱讀 3263·2019-08-29 16:30