1. 程式人生 > >Object instanceof Function 還是 Function instance of Object,是真是假,一一道來

Object instanceof Function 還是 Function instance of Object,是真是假,一一道來

如今的JavaScript再也不是以前被當做玩具的在網頁上執行的花哨的指令碼了。JavaScript已經逐漸標準化,作為一門真正的程式語言廣泛地應用在Web開發上。因此,越來越多的人開始重新認識這門指令碼語言,並在不斷地探索關於JavaScript核心思想和實現原理,過程中遇到了一些非常混淆的問題。本文著重解釋一個比較常見但是非常容易使開發人員或者是初學JavaScript的人非常混淆的問題,那就是兩個核心建構函式Object和Function,他們之間到底有什麼關係?為何instanceof運算子的返回結果會讓你感到混淆?本文將為你一一道來。不過在這之前,我們需要先了解一些JavaScript中的概念和基本的執行機制。

JavaScript的物件體系結構

其實在JavaScript語言中,整個核心的體系結構都圍繞著兩個建構函式Object和Function來構建的。我將引用來自mollypages.org的一張JavaScript物件體系結構圖來說明。

圖1-1 The JavaScript Object Hierarchy

instanceof 運算子

instanceof是一個二元運算子,如:A instanceof B. 其中,A必須是一個合法的JavaScript物件,B必須是一個合法的JavaScript函式 (function). 判斷過程如下:

  如果函式B在物件A的原型鏈 (prototype chain) 中被發現,那麼instanceof操作符將返回true,否則返回false

.

例如下面的程式碼會返回true.

// return true if specified function is found
//
in the object's prototype chain as a constructor.alert({} instanceof Object);

JavaScript中的原型鏈(prototype chain)機制

這裡簡單概括一下,因為這個話題需要很大篇幅去討論,本文只是引用了這個概念,重點並非詳細討論該機制。

JavaScript中的原型(prototype)是和函式(function)緊密相連的,因為每個函式預設都會有一個屬性叫prototype, 每一個通過函式和new操作符生成的物件都具有一個屬性__proto__, 這個屬性儲存了建立它的建構函式的prototype屬性的引用。這個__proto__物件就是實現原型鏈的核心物件。JavaScript是一門面向物件的程式語言,它的繼承特性其實就是通過原型鏈機制來實現的。同時,instanceof運算子也需要在原型鏈的支援。我們舉例說明:

複製程式碼 程式碼 // create a custom constructor Foofunction Foo() {
}
// create an insatnce of Foovar foo =new Foo();

// foo is an instance of Fooalert(foo instanceof Foo);// true
//
foo is also an instance of Object because
//
Foo.prototype is an instance of Object.
//
the interpreter will find the constructor
//
through the prototype chain.alert(foo instanceof Object);// true// Prototype chain of the object foo
//
//
__proto__ __proto__ __proto__
//
foo -----------> Foo.prototype -----------> Object.prototype -----------> null// But foo is not an instance of Function, because
//
we could not find Function.prototype in foo's
//
prototype chain.alert(foo instanceof Function);// false// However, its constructor Foo is an instance of
//
Function.alert(Foo instanceof Function);// true
//
it's also an instance of Object alert(Foo instanceof Object);// true// Prototype chain of the constructor Foo
//
//
__proto__ __proto__ __proto__
//
Foo -----------> Function.prototype -----------> Object.prototype -----------> null 複製程式碼

從上面的程式碼來分析,我們不難得出這樣一個結論:任何物件的原型鏈最後都能追溯到Object.prototype. 這也就是我們為什麼說JavaScript中所有的物件都繼承自Object的原因了。

為何Object instanceof Function和Function instanceof Object都返回true?

Object, Function, Array等等這些都被稱作是構造“函式”,他們都是函式。而所有的函式都是建構函式Function的例項。從原型鏈機制的的角度來說,那就是說所有的函式都能通過原型鏈找到建立他們的Function建構函式的構造原型Function.protorype物件,所以:

alert(Object instanceof Function);// return true 與此同時,又因為Function.prototype是一個物件,所以他的建構函式是Object. 從原型鏈機制的的角度來說,那就是說所有的函式都能通過原型鏈找到建立他們的Object建構函式的構造原型Object.prototype物件,所以: alert(Function instanceof Object);// return true 有趣的是根據我們通過原型鏈機制對instanceof進行的分析,我們不難得出一個結論:Function instanceof Function 依然返回true, 原理是一樣的

1. Function是建構函式,所以它是函式物件

2. 函式物件都是由Function建構函式建立而來的,原型鏈機制解釋為:函式物件的原型鏈中存在Function.prototype

3. instanceof查詢原型鏈中的每一個節點,如果Function.prototype的建構函式Function的原型鏈中被查到,返回true

因此下面程式碼依然返回true

alert(Function instanceof Function);// still true

結論

1. 在JavaScript語言中,一切的一切都是物件,它們全部繼承自Object. 或者說所有物件的原型鏈的根節點都是Object.prototype

2. 理解原型鏈機制在JavaScript中式如何工作的是非常重要的。掌握了它,不管一個物件多麼複雜,你總能夠輕而易舉地將它攻破。