在進入正文以前,我得先說說我認識js的prototype這個東西的曲折過程。html
百度js的prototype的文章,先看看,W3School關於prototype的介紹:數組
你以爲這概念適合定義js的prototype這個東西嗎?你是否也認爲prototype是一個object對象的屬性呢?是的話,請認真認真看我這篇文章,由於這篇文章會毀滅你的人生三觀,呵呵,就是有這麼嚴重,由於本人就是被這個定義給害慘的。函數
不得不說,看了網上的一些介紹prototype的文章,基本上都說prototype是對象的一個屬性,因而,我也堅決的認爲prototype是一個對象的屬性,因此,我被了坑很久很久,由此,引起的後果就是,我一次次的誤解別人寫的含有prototype的js代碼,也就是當別人亮出js的prototype這個屬性來寫js代碼時,我看着他們寫的代碼都是 ····一頭霧水····· ,悲催啊!因此,我恨死prototype這個東西了,所以,在這裏,我今天必須把js的prototype屬性道個明明白白。看官,請擦亮你的眼睛,仔細看我下面的實驗。測試
固然,我但願諸位看官也可以靜下心來,把我下面的實驗從新作一遍,好證實個人結論是正確的。spa
同時,也爲了證實·····我沒有····騙大家·····,呵呵,廢話很少說了,下面進入實驗階段。prototype
先介紹一個下面要用到的函數,JSON.stringify(value)。3d
這個函數的做用是:把傳入的參數value變成字符串,它有三個參數,第一個參數是必須的,其他的兩個參數可填可不填。code
關於JSON.stringify函數的做用請看這篇文章。http://www.cnblogs.com/ningvsban/p/3660654.html,這裏說的很清楚。htm
首先,測試W3Schol的定義到底行不行的通:對象
若是,真如W3Schol所說的那樣,prototype是object的一個屬性,那麼,咱們來建立一個對象,看看這個對象的prototype究竟是啥。
var ob = { };//超級簡單的空對象
alert(JSON.stringify(ob.prototype));
你以爲上面的代碼會alert出什麼呢?既然prototype是object的一個屬性,那麼上面確定可以獲取到什麼東西對吧?可是,若是你拿這段代碼去作實驗了,你會被打臉的,由於它alert的東西是·········undefined··········,也就是說object這個屬性prototype不是個東西,很殘酷吧,但現實就是這樣,任何對象引用prototype都會出現undefined,不信,你能夠試一試。
我能夠很明白的告訴你,prototype絕對不是給對象用的,對象根本沒辦法引用prototype這個屬性,它真正的屬主,實際上是···········函數········,記住,可以引用prototype的東西絕對是函數,絕對是函數,絕對是函數,prototype是屬於函數的一個屬性,prototype是屬於函數的一個屬性,prototype是屬於函數的一個屬性,可以引用它的只有·····函數····,可以引用它的只有·····函數·····,可以引用它的只有····函數····,函數,函數,函數,重要的事情必定要說千百遍,呵呵,由於只有明確這一點,下面,你才明白我要講的東西。別怪我這麼墨跡啊!
下面,我要給prototype一個名副其實的定義:
prototype是函數的一個屬性,是函數的原型對象。
就是這麼簡單,能看明白嗎? prototype只可以被 函數 調用。
別搞混了js的object對象和function函數,js的對象和函數絕對是兩個概念,爲何?由於js的function函數能夠new出object對象啊,是吧?這個你總該知道吧?
下面,來作試驗證實個人結論。
//首先定義一個有名函數func
function func(){ } alert(func.prototype);
你說,上面的代碼會alert什麼呢?還會不會是undefined的呢?我敢確定的告訴你絕對不是undefined的,由於本人已經alert過了,呵呵。
上面,彈出的窗口是:
沒錯,返回的就是對象,這回你肯相信我,可以調用prototype的必定是函數了吧?object引用prototype的時候給你返回的是一個不是東西的東西undefined,函數引用prototype的時候給你返回一個真真實實存在的東西object的,這還不夠證實prototype是給函數用的嗎?還不可以證實對象是不能引用prototype的嗎?呵呵,又多說了。
上面我已經說過,prototype是函數的一個屬性,也是函數的原型對象,而這裏func函數引用prototype的時候返回的是一個對象object的,那麼,結合這兩個概念,你能得出什麼結論呢?我想經過這不可貴出結論:
prototype是········函數的原型對象············,這能理解嗎?不能理解,沒有關係,咱們再來作一個實驗,終於要用到文章開始給大家介紹的一個函數JSON.stringify()了。
function func(){ } alert(JSON.stringify(func.prototype));
仍是引用上面的函數func,只不過這裏返回的是JSON.stringify()函數的返回值。
你沒有看錯,這裏alert的結果是一個空對象,這證實,prototype的的確確屬於函數的屬性,而且函數的prototype屬性的js數據類型是對象,明白不?爲何如今是一個空對象?你有沒有想過?爲何呢?請思考這個問題三秒鐘,假如想不出來,那麼沒有關係,下面,我來解釋。
先看,實驗代碼:
function func(){ } func.prototype.name ='prototype是函數的的屬性,本質是函數的原型對象'; alert(JSON.stringify(func.prototype))
你說,上面的代碼會alert出什麼呢?請看下面:
你沒有看錯,此刻,終於alert出東西來了,我在這裏給prototype賦於一個屬性name,因此,我這個時候在alert函數func的prototype時,你看到有值了嗎?看到了prototype的屬性name值了嗎?如今再想一想,爲何上面第一層alert函數func.prototype的時候,它是一個空對象呢?而如今它又是一個有值的對象呢?
緣由很簡單呢,由於第一次的時候,我沒有給函數的屬性func.prototype賦予name屬性,也沒有給name屬性賦值啊,而如今我已經給函數的屬性func.prototype賦予屬性name,而且賦值爲········prototype是函數的的屬性,本質是函數的原型對象······,因此,如今alert函數的屬性func.prototype的時候它就有值啦,對吧?
所以,這裏得出結論:
prototype是函數的的屬性,本質是函數的原型對象。
別覺得js中只有對象纔有屬性,經過這裏,咱們也能夠知道,其實js的函數也是有屬性的,並且js的函數好像就只有這個屬性prototype,並且js的這個函數屬性同時仍是函數的原型對象,你是否是被搞暈了?但願你沒有被搞暈纔好。
爲何說prototype是函數的一個屬性呢?由於,只有函數才能調用prototype,並且是以這樣的方式func.prototype調用的,這樣的方式調用東西是否是和對象調用屬性如出一轍呢?是的,就是由於函數調用prototype的時候是和對象調用屬性的時候同樣的,咱們才把prototype說成是函數的一個屬性,而函數的這個屬性實際上是一個對象(這個是否是對象,上面已經證實了,這裏就再也不說明),因此說,這個prototype就是函數的屬性,本質是函數的原型對象。
這裏爲何強調說prototype的本質是函數的原型對象呢?
下面看代碼證實,個人代碼很簡單的:
//定義一個函數
function func(){ } //給函數的屬性prototype賦予一個方法get
func.prototype.get=function(value){ return value;//很簡單,你給我什麼我就輸出什麼
}
你說,怎麼調用上面那個get方法?
我給你一點提示,get是屬於func函數的一個屬性函數,既然是屬性函數,那麼咱們怎麼調用呢?
很簡單,屬性函數必須由它的對象來調用,那麼咱們怎麼獲取get的對象呢?很簡單,用關鍵字new來實例化func函數的對象就好了嗎?是吧?
下面,實例化func函數的一個對象ob1:
var ob1 = new func(); //用func實例化出來的對象來調用get屬性函數 alert(ob1.get('hello,prototype原型對象'));
沒錯,用func函數實例化出來的對象ob1,確實可以調用get函數,上面已經利用ob1調用get函數alert出來了,這就證實func函數的實例對象是擁有get這個屬性函數的,對吧?這麼明顯,就不用說明了吧。
若是,你不信必需要用實例化func的對象來調用get函數,你能夠試試直接用func調用get函數,也就是說,你能夠嘗試着func.get()調用一下get函數看看,看func它到底能不可以直接調用get函數。我想必定報錯,呵呵,緣由是什麼,你本身想。
那麼,說了這麼多,我仍是沒有說明,爲何我說prototype的本質是函數的原型對象啊?對吧?
請看,下面的代碼:
var ob2 = new func(); //用func實例化出來的對象來調用get屬性方法 alert(ob2.get('我依然是func實例化出來的對象'));
看到沒有,我仍是再次利用func函數實例化一個ob2,這個對象依然可以調用get屬性函數,這說明什麼了呢?這說明了,利用func對象實例化的全部對象均可以調用get這個屬性函數。既然如此,你說,prototype這個屬性若是不是函數func的原型對象的話,那麼爲何給prototype賦予的屬性函數get可以被func全部實例化的對象所調用呢?假如prototype的本質不是func函數的原型對象,那麼依據func函數實例化出來的對象就不可能一個個都可以調用get這個屬性方法,對吧?爲何?咱們來個比例,你說,一個不是源頭的東西,它能影響到全部的東西嗎?不可以吧?所以,這裏得出結論:
prototype是函數的一個屬性,本質就是函數的原型對象。
整篇文章就是爲了說明這個結論。無他。但願看官能看懂這篇文章。對於prototype的應用下篇文章再講,在此打住。
特別指出:
Array.prototype是一個數組
String.prototype是一個字符串
Object.prototype是一個對象
這三個特殊例子,不像構造函數的prototype同樣。詳情請看下一篇文章。