摘要:構(gòu)造函數(shù),實例,原型三者的關(guān)系如下圖構(gòu)造函數(shù)是構(gòu)成整個原型鏈的關(guān)鍵,是他利用將原型傳給了后代。因此,通過操縱構(gòu)造函數(shù)的,就能夠操縱原型鏈,從而對原型鏈進行自在的拼接。
要理解js的原型鏈主要就是理清楚以下三者的關(guān)系:
構(gòu)造函數(shù)的protitype屬性
對象的__proto__屬性
對象的constructor屬性
在js中,函數(shù)作為一等公民,它是一個對象,可以擁有自己的屬性,可以像其他類型(比如string)一樣作為參數(shù)進行傳遞,作為返回值進行返回。
我們首先寫一個名為Animal的函數(shù)。我們將從這個Animal函數(shù)開始,通過原型鏈,構(gòu)建一條“動物-狗-警犬”的繼承關(guān)系。
function Animal() {}
首先把目光放在這個Animal函數(shù)身上。這是一個普通的空函數(shù),當函數(shù)被關(guān)鍵字new調(diào)用時,我們就說他此刻是作為一個構(gòu)造函數(shù),通常用首字母大寫表示。
接下來實例化這個Animal類
var animal = new Animal();
這時,js發(fā)生了以下過程:
從內(nèi)存中獲取一個空對象{}交給animal。
設(shè)置animal.__proto__為Animal.prototype
執(zhí)行構(gòu)造函數(shù)中的其他操作,比如利用this設(shè)置animal的屬性等,因為此時this指向的是animal。我們這里是空函數(shù),因此沒有操作。
打印animal到控制臺。
可以看到animal只有一個__proto__屬性,并且在__proto__中還有一個__proto__屬性,這就是一條原型鏈,由__proto__組成的原型鏈。
當在自身屬性上沒有查找到某個屬性時,js就會嘗試查找__proto__上有無該屬性,不斷的向著上級__proto__爬,一直到找到那個屬性為止,或者沒有找到,返回undefined。
animal的第二級__proto__指向的就是Object。因為所有的對象的原型鏈頂端都是Object。
在_propto_屬性中我們還能看到constructor屬性,該屬性指向一個構(gòu)造函數(shù)。他們之間的關(guān)系即是:構(gòu)造函數(shù)的prototype屬性指向了原型,而原型上的constructor又回指構(gòu)造函數(shù)Animal。
構(gòu)造函數(shù),實例,原型三者的關(guān)系如下圖:
構(gòu)造函數(shù)Animal是構(gòu)成整個原型鏈的關(guān)鍵,是他利用prototype將原型傳給了后代。因此,通過操縱構(gòu)造函數(shù)的prototype,就能夠操縱原型鏈,從而對原型鏈進行自在的拼接。
現(xiàn)在我們就開始打造Animal->Dog->PoliceDog的原型鏈。
Animal.prototype.breathe = function() { console.log("breathe~"); } function Dog(name){ this.name = name; } function PoliceDog(name,id) { Dog.call(this, name); this.id = id; }
我們在Animal的prototype上設(shè)置了一個breathe方法,之后又新建了Dog和PoliceDog這兩個構(gòu)造函數(shù),在構(gòu)造函數(shù)中利用call和this設(shè)置了一些自己獨有的屬性。
現(xiàn)在, Animal,Dog,PoliceDog彼此還沒有交集。當然他們的頂級原型都是Object。
Dog.prototype = new Animal(); PoliceDog.prototype = new Dog();
上兩句代碼就將三者的原型攛在了一起。接下來還需要將原型上的constructor進行回指。
Dog.prototype.constructor = Dog; PoliceDog.Prototype.constructor = PoliceDog;
下面繼續(xù)為Dog和PoliceDog添加一些方法
Dog.prototype.constructor = Dog; Dog.prototype.bark = function(){ console.log("汪!汪!"); } PoliceDog.prototype.manhunt = function() { console.log(this.name + "向犯罪份子瘋狂發(fā)動進攻!") } PoliceDog.prototype.checkId = function() { console.log("警犬"+this.name +"的id是:"+this.id); }
此時我們new一個PoliceDog,并將其打印至控制臺。
可以清晰的看到這條原型鏈。
但是我們也注意到在policeDog的第二級原型上繼承了name屬性,但該屬性我們已經(jīng)在構(gòu)造函數(shù)中為自己設(shè)置了,我們不希望在原型上也繼承該屬性。
會出現(xiàn)這種情況的原因是因為我們在拼接原型時用的是new出來的一個實例。Dog的實例中存在name屬性。
因此可以采用另一種拼接方法。
PoliceDog.prototype = Object.create(Dog.prototype);
Object.create函數(shù)接受一個對象A,并返回一個對象,返回的對象的__proto__為對象A。
形如這個樣子:Object.create(對象A),返回{__proto__:對象A}
關(guān)于Object.create具體的請移步這里。
修改之后再次打印policeDog的實例
可以看到name屬性沒有出現(xiàn)在原型里。
使用es6實現(xiàn)繼承,底層也是使用原型鏈
class Animal { breathe() { console.log("breathe") }; } class Dog extends Animal { constructor(name){ super(); this.name = name; } bark(){ console.log("汪!汪!"); } } class PoliceDog extends Dog { constructor(name,id){ super(name); this.id = id; } manhunt(){ console.log(this.name + "向犯罪份子瘋狂發(fā)動進攻!"); } checkId(){ console.log("警犬"+this.name +"的id是:"+this.id); } } // 測試 let policeDog_1 = new PoliceDog("peter","k0302"); console.log(policeDog_1);
與我們使用Object.create修改原型鏈達到一樣的效果
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/95375.html
摘要:原型鏈和對象的原型是對象實例和它的構(gòu)造函數(shù)之間建立的鏈接,它的值是構(gòu)造函數(shù)的。對象的原型根據(jù)上文提到的構(gòu)造調(diào)用函數(shù)的時候會創(chuàng)建一個新對象,自動將的原型指向構(gòu)造函數(shù)的對象。 showImg(https://segmentfault.com/img/remote/1460000020185197); JS的原型、原型鏈一直是比較難理解的內(nèi)容,不少初學者甚至有一定經(jīng)驗的老鳥都不一定能完全說清...
摘要:從實現(xiàn)角度分析原型鏈歡迎來我的博客閱讀從實現(xiàn)角度分析原型鏈網(wǎng)上介紹原型鏈的優(yōu)質(zhì)文章已經(jīng)有很多了,比如說作為補充,就讓我們換個角度,從實現(xiàn)來分析一下吧本文假設(shè)你對原型鏈已經(jīng)有所了解。 從實現(xiàn)角度分析js原型鏈 歡迎來我的博客閱讀:《從實現(xiàn)角度分析js原型鏈》 網(wǎng)上介紹原型鏈的優(yōu)質(zhì)文章已經(jīng)有很多了,比如說: https://github.com/mqyqingfeng/Blog/issu...
摘要:相當于在用原型繼承編寫復雜代碼前理解原型繼承模型十分重要。同時,還要清楚代碼中原型鏈的長度,并在必要時結(jié)束原型鏈,以避免可能存在的性能問題。 js是一門動態(tài)語言,js沒有類的概念,ES6 新增了class 關(guān)鍵字,但只是語法糖,JavaScript 仍舊是基于原型。 至于繼承,js的繼承與java這種傳統(tǒng)的繼承不一樣.js是基于原型鏈的繼承. 在javascript里面,每個對象都有一...
摘要:圖片描述缺點是無法實現(xiàn)多繼承可以在構(gòu)造函數(shù)中,為實例添加實例屬性。 對象的方法 Object.assign() 對象可以簡寫 ,如果 key 和 value 相等則可以簡寫 let name = xm; let age = 2; let obj = { name, age, fn(){ // 可以省略函數(shù)關(guān)鍵字和冒號: console.log(2...
摘要:二構(gòu)造函數(shù)我們先復習一下構(gòu)造函數(shù)的知識上面的例子中和都是的實例。這兩個實例都有一個構(gòu)造函數(shù)屬性,該屬性是一個指針指向。原型鏈其中是對象的實例。 一. 普通對象與函數(shù)對象 JavaScript 中,萬物皆對象!但對象也是有區(qū)別的。分為普通對象和函數(shù)對象,Object 、Function 是 JS 自帶的函數(shù)對象。下面舉例說明 var o1 = {}; var o2 =new Objec...
閱讀 476·2021-10-09 09:57
閱讀 483·2019-08-29 18:39
閱讀 820·2019-08-29 12:27
閱讀 3036·2019-08-26 11:38
閱讀 2674·2019-08-26 11:37
閱讀 1300·2019-08-26 10:59
閱讀 1387·2019-08-26 10:58
閱讀 996·2019-08-26 10:48