最開始想要直觀的理解請(qǐng)直接參考阮一峰老師博客關(guān)于繼承的講解:http://www.ruanyifeng.com/blog/2011/06/designing_ideas_of_inheritance_mechanism_in_javascript.html
原型對(duì)象
javascript語言是一種面向?qū)ο蟮恼Z言饮六,它沒有"子類"和"父類"的概念嫡良,里面所有的數(shù)據(jù)類型都是對(duì)象,如何將這些對(duì)象聯(lián)系起來呢?
Brendan Eich在考慮設(shè)計(jì)繼承機(jī)制的時(shí)候哩牍,參考了C++和JAVA使用new命令,通過調(diào)用類的構(gòu)造函數(shù)生成實(shí)例的方式,將new命令引入javascript。
C++的寫法是:
ClassName *object = new ClassName(param);
Java的寫法是:
Person person = new Person();
但是乱灵,javascript里面沒有“類”這個(gè)概念,那么七冲,Brendan Eich決定直接在new后面跟一個(gè)構(gòu)造函數(shù)痛倚,來生成實(shí)例。
構(gòu)造函數(shù)是什么澜躺?構(gòu)造函數(shù)與其他函數(shù)唯一的區(qū)別在于調(diào)用方式不同蝉稳。任何函數(shù)只要通過new來調(diào)用就可以作為構(gòu)造函數(shù),它是用來創(chuàng)建特定類型的對(duì)象掘鄙。
下面定義一個(gè)構(gòu)造函數(shù)Female:
function Female(name){
this.name = name;
this.sex = 'female';
}
通過new命令來生成一個(gè)person實(shí)例:
var person1 = new Female("Summer")
這里耘戚,構(gòu)造函數(shù)Female就是實(shí)例對(duì)象person1的原型!2倌收津!Female里的this關(guān)鍵字就指的是person1這個(gè)對(duì)象!
new出來的person1對(duì)象此時(shí)已經(jīng)和Female再無聯(lián)系了浊伙!也就是說每一個(gè)new出來的實(shí)例都有自己的屬性和方法的副本撞秋,是獨(dú)立的的!修改其中一個(gè)不會(huì)影響另一個(gè)嚣鄙!
var person1 = new Female("Summer");
var person2 = new Female("Lily");
person2.sex = 'male';
console.log(person1.sex) // female
console.log(person2.sex) // male
但是吻贿,我們希望構(gòu)造函數(shù)中的sex屬性是一個(gè)共有屬性,那么此時(shí)用這樣的方法拗慨,每個(gè)實(shí)例中都有一個(gè)相同的sex屬性廓八,會(huì)造成資源極大的浪費(fèi)奉芦!
那么原型對(duì)象就即將登場(chǎng)了赵抢!Brendan Eich決定給每一個(gè)構(gòu)造函數(shù)都設(shè)置一個(gè)prototype屬性,這個(gè)屬性就指向原型對(duì)象声功。其實(shí)原型對(duì)象就只是個(gè)普通對(duì)象烦却,里面存放著所有實(shí)例對(duì)象需要共享的屬性和方法!所以先巴,我們把需要共享的放到原型對(duì)象里其爵,把那些不需要共享的屬性和方法存在在構(gòu)造函數(shù)里!
那么上面的代碼可改寫如下:
function Person(name,age){ this.name = name;
}
Person.prototype.sex = 'female';
var person1 = new Person("Summer");
var person2 = new Person("Lily");
console.log(person1.sex) // female
console.log(person2.sex) // female
Person.prototype.sex = 'male';
console.log(person1.sex) // male
console.log(person2.sex) // male</pre>
可以看出伸蚯,修改prototype屬性會(huì)影響它的所有實(shí)例的sex的值Dγ臁!
實(shí)例一旦創(chuàng)建出來就會(huì)自動(dòng)引用prototype對(duì)象的屬性和方法剂邮!所以實(shí)例對(duì)象的屬性和方法一般分為兩種:一種是自身的摇幻,一種是引用自prototype的。
具體實(shí)現(xiàn)是這樣的:
每當(dāng)代碼讀取某個(gè)對(duì)象的某個(gè)屬性的時(shí)候,都會(huì)執(zhí)行一次搜索绰姻。首先從對(duì)象實(shí)例本身開始枉侧,如果在實(shí)例中找到了該屬性,則返回該屬性的值狂芋,如果沒有找到榨馁,則順著原型鏈指針向上,到原型對(duì)象中去找帜矾,如果如果找到就返回該屬性值翼虫。
這里要提一點(diǎn),如果為對(duì)象實(shí)例添加了一個(gè)屬性與原型中同名黍特,則該屬性會(huì)屏蔽掉原型中的同名屬性蛙讥,不會(huì)去修改它!使用delete可以刪除實(shí)例中的屬性(提到delete那要插一句delete只能刪除對(duì)象下的屬性灭衷,不能刪除變量和參數(shù)次慢!)
原型鏈
事實(shí)上,js里完全依靠"原型鏈"(prototype chain)模式來實(shí)現(xiàn)繼承翔曲。
上面說完原型對(duì)象迫像。下面要扒一扒proto、prototype瞳遍、constructor
proto:事實(shí)上就是原型鏈指針N偶恕!
prototype:上面說到這個(gè)是指向原型對(duì)象的
constructor:每一個(gè)原型對(duì)象都包含一個(gè)指向構(gòu)造函數(shù)的指針掠械,就是constructor
繼承實(shí)現(xiàn)方式:
為了實(shí)現(xiàn)繼承由缆,proto會(huì)指向上一層的原型對(duì)象,而上一層的結(jié)構(gòu)依然類似猾蒂,那么就利用proto一直指向Object的原型對(duì)象上均唉!Object.prototype.proto = null;表示到達(dá)最頂端。如此形成了原型鏈繼承肚菠。
下面有個(gè)圖解非常經(jīng)典舔箭,我自己也手畫了一遍去理解,真的非常有效~
大致總結(jié)一下就是:
Object是作為眾多new出來的實(shí)例的基類 function Object(){ [ native code ] }
Function是作為眾多function出來的函數(shù)的基類 function Function(){ [ native code ] }
構(gòu)造函數(shù)的proto(包括Function.prototype和Object.prototype)都指向Function.prototype
原型對(duì)象的proto都指向Object.prototype
Object.prototype.proto指向null
轉(zhuǎn)自【http://www.cnblogs.com/ningyn0712/p/6216711.html】