|
js 代碼
復(fù)制代碼 代碼如下:
var animal = function (){ //這就是constructor function 了
this .name = 'pipi';
this .age = 10;
this .height = 0;
}
//建立一個(gè)動(dòng)物的實(shí)例
var a1 = new animal ();
構(gòu)造函數(shù)與其他普通函數(shù)區(qū)別在于,1.構(gòu)造函數(shù)里有 this關(guān)鍵字,2.調(diào)用構(gòu)造函數(shù)是使用的new關(guān)鍵字。通過new運(yùn)算符調(diào)用構(gòu)造函數(shù) animal 后,系統(tǒng)就會(huì)返回一個(gè)對(duì)象,這個(gè)對(duì)象就相當(dāng)于
復(fù)制代碼 代碼如下:
var a1 = { name:'pipi' ,age:10,height:0 }
//或者
var a1 = new Object();
a1.name='pipi';
a1.age = 10;
a1.height = 0;
等同這樣的方式來產(chǎn)生js對(duì)象。
到這里我們知道如何在js中定義一個(gè)類了,接下來我們展示如何寫一個(gè)cat
復(fù)制代碼 代碼如下:
var cat = function (){
this .play = function (){
alert('cat play')
}
}
cat .prototype = new animal ();
//prototype 屬性指向一個(gè)對(duì)象
var c1 = new cat();
到這里,cat就繼承了 animal 對(duì)象,類cat的一個(gè)實(shí)例對(duì)象c1擁有屬性name,age,height,和方法play了。
那么 prototype起到了一個(gè)什么樣的作用呢?
prototype就好比一個(gè)指針,它指向一個(gè)object,這個(gè)object就稱為子類對(duì)象的原型。當(dāng)cat的對(duì)象被創(chuàng)建的時(shí)候,由于cat的構(gòu)造函數(shù)擁有prototype屬性,那么cat的實(shí)例就會(huì)間接指向這個(gè)原型對(duì)象了(說成間接的是因?yàn)槊總€(gè)object都有一個(gè) constructor 屬性指向它的構(gòu)造函數(shù))。
那么問題來了,“當(dāng)我們修改對(duì)象 c1 的name屬性的時(shí)候,會(huì)不會(huì)修改它prototype的name屬性值呢?”,答案是否定的。
接下來詳細(xì)解析:
1.訪問name屬性: 首先當(dāng)我們第一次訪問c1.name的屬性的時(shí)候,我們會(huì)得到值“pipi”,這個(gè)和我們預(yù)料中的一樣。但是計(jì)算過程你未必知道。
它計(jì)算的過程是這樣的:第一步:檢查c1對(duì)象中是否有name屬性,找到的話就返回值,沒有就跳到第二步,顯然沒有找到,因?yàn)閏at的構(gòu)造函數(shù)中沒有定義。第二步:當(dāng)?shù)谝徊經(jīng)]有找時(shí),去間接訪問 prototype對(duì)象所指向的object,如果在 prototype對(duì)象中 找到的name屬性的話,就返回找到的屬性值。如果還是沒有找到的話,再去遞歸地尋找 prototype對(duì)象的 prototype對(duì)象(去找它的爺爺) ,一直到找到name屬性或者沒有prototype對(duì)象為止。如果到最后還是沒有找到name屬性的話就返回undefined。
2.設(shè)定name屬性:當(dāng)我們?cè)O(shè)定c1對(duì)象的name屬性時(shí),及調(diào)用 c1.name= ' new name'; 這個(gè)過程就簡(jiǎn)單多了。首先檢查是否對(duì)象已有該屬性,若已存在則修改當(dāng)前值,若不存在則為該對(duì)象新增一個(gè)屬性并設(shè)定當(dāng)前值。值得一提的是,在設(shè)定值的過程中沒有去訪問 prototype屬性。
為了加深理解,我們?cè)倏匆粋€(gè) read-write-read 的過程,第一次read的時(shí)候,由于自己的對(duì)象沒有name屬性,那么就會(huì)返回的原型對(duì)象的name屬性的值。第二步,寫入name的值,同樣沒發(fā)現(xiàn)本身對(duì)象有name屬性,那么就在本身對(duì)象上新建一個(gè)name屬性,然后賦值。第三步,再次讀取name屬性,由于在第二步中已經(jīng)新建了name屬性,此時(shí)就返回在第二步中設(shè)定的值。值得一提的是,在這三步中沒有改變?cè)蛯?duì)象的值。
好了,到此詳細(xì)分析了 Javascript對(duì)象是如果實(shí)現(xiàn)繼承的,其實(shí)和其他的面向?qū)ο笳Z言不一樣的是,Javascript的繼承機(jī)制是對(duì)象的原型繼承而不是類型繼承。
呵呵,歡迎看完,有不對(duì)的地方歡迎大家討論!
JavaScript技術(shù):javascript類繼承機(jī)制的原理分析,轉(zhuǎn)載需保留來源!
鄭重聲明:本文版權(quán)歸原作者所有,轉(zhuǎn)載文章僅為傳播更多信息之目的,如作者信息標(biāo)記有誤,請(qǐng)第一時(shí)間聯(lián)系我們修改或刪除,多謝。