一張圖理解JS的原型(prototype、_proto_、constructor的三角關係)

注意:前方高能預警,請認真仔細看完,閱讀完後本身再次畫下原型圖,相信你必定會有更深入的認識。(推薦炒雞好用的畫流程圖的軟件ProcessOn)

構造函數:function Foo ( ) { };bash

實例對象:let f1=new Foo;函數

let o1=new Foo;spa

每一個函數都有 prototype 屬性,除了 Function.prototype.bind(),該屬性指向原型prototype

每一個對象都有 __proto__ 屬性指向了建立該對象的構造函數的原型。其實這個屬性指向了 [[prototype]],可是 [[prototype]] 是內部屬性,咱們並不能訪問到,因此使用 _proto_ 來訪問。3d

對象能夠經過 __proto__ 來尋找不屬於該對象的屬性,__proto__ 將對象鏈接起來組成了原型鏈。code

下面咱們來解釋上圖的原型圖的含義:cdn

概念:對象

一、構造函數:用來初始化新建立的對象的函數是構造函數。在例子中,Foo()函數是構造函數。blog

二、實例對象:經過構造函數的new操做建立的對象是實例對象。能夠用一個構造函數,構造多個實例對象。繼承

function Foo(){};
var f1 = new Foo;
var f2 = new Foo;
console.log(f1 === f2);//false複製代碼

三、原型對象及prototype:構造函數有一個prototype屬性,指向實例對象的原型對象。經過同一個構造函數實例化的多個對象具備相同的原型對象。常常使用原型對象來實現繼承。

function Foo(){};
Foo.prototype.a = 1;
var f1 = new Foo;
var f2 = new Foo;

console.log(Foo.prototype.a);//1
console.log(f1.a);//1
console.log(f2.a);//1複製代碼

四、constructor:原型對象有一個constructor屬性,指向該原型對象對應的構造函數。因爲實例對象能夠繼承原型對象的屬性,因此實例對象也擁有constructor屬性,一樣指向原型對象對應的構造函數。

console.log(Foo.prototype.constructor === Foo);//true
console.log(f1.constructor === Foo);//true複製代碼

五、_proto_:實例對象有一個proto屬性,指向該實例對象對應的原型對象。

console.log(f1.__proto__ === Foo.prototype);//true複製代碼

概念介紹完了,如今對圖示的關係進行詳細說明

【第一部分: Foo】

一、實例對象f1是經過構造函數Foo()的new操做建立的。構造函數Foo()的原型對象是Foo.prototype;實例對象f1經過__proto__屬性也指向原型對象Foo.prototype。

console.log(f1.__proto === Foo.prototype);//true
複製代碼

二、實例對象f1自己並無constructor屬性,但它能夠繼承原型對象Foo.prototype的constructor屬性

console.log(Foo.prototype.constructor === Foo);//true
console.log(f1.constructor === Foo);//true
console.log(f1.hasOwnProperty('constructor'));//false複製代碼

  下圖是實例對象f1的控制檯效果

【第二部分: Object】

一、Foo.prototype是f1的原型對象,同時它也是實例對象。實際上,任何對象均可以看作是經過Object()構造函數的new操做實例化的對象 因此,Foo.prototype做爲實例對象,它的構造函數是Object(),原型對象是Object.prototype。相應地,構造函數Object()的prototype屬 性指向原型對象Object.prototype;實例對象Foo.prototype的proto屬性一樣指向原型對象Object.prototype。

console.log(Foo.prototype.__proto__ === Object.prototype);//true
複製代碼

二、實例對象Foo.prototype自己具備constructor屬性,因此它會覆蓋繼承自原型對象Object.prototype的constructor屬性。

console.log(Foo.prototype.constructor === Foo);//true
console.log(Object.prototype.constructor === Object);//true
console.log(Foo.prototype.hasOwnProperty('constructor'));//true複製代碼

  下圖是實例對象Foo.prototype的控制檯效果

三、若是Object.prototype做爲實例對象的話,其原型對象是什麼,結果是null。我覺得,這可能也是typeof null的結果是'object'的緣由之一吧。

console.log(Object.prototype.__proto__ === null);//true
複製代碼

【第三部分: Function】

一、前面已經介紹過,函數也是對象,只不過是具備特殊功能的對象而已。任何函數均可以看作是經過Function()構造函數的new操做實例化的結果。若是把函數Foo當成實例對象的話,其構造函數是Function(),其原型對象是Function.prototype;相似地,函數Object的構造函數也是Function(),其原型對象是Function.prototype。

console.log(Foo.__proto__ === Function.prototype);//true
console.log(Object.__proto__ === Function.prototype);//true複製代碼

二、原型對象Function.prototype的constructor屬性指向構造函數Function();實例對象Object和Foo自己沒有constructor屬性,須要繼承原型對象Function.prototype的constructor屬性。

console.log(Function.prototype.constructor === Function);//true
console.log(Foo.constructor === Function);//true
console.log(Foo.hasOwnProperty('constructor'));//false
console.log(Object.constructor === Function);//true
console.log(Object.hasOwnProperty('constructor'));//false
複製代碼

三、全部的函數均可以當作是構造函數Function()的new操做的實例化對象。那麼,Function能夠當作是調用其自身的new操做的實例化的結果。因此,若是Function做爲實例對象,其構造函數是Function,其原型對象是Function.prototype。

console.log(Function.__proto__ === Function.prototype);//true
console.log(Function.prototype.constructor === Function);//true
console.log(Function.prototype === Function.prototype);//true複製代碼

四、若是Function.prototype做爲實例對象的話,其原型對象是什麼呢?和前面同樣,全部的對象均可以當作是Object()構造函數的new操做的實例化結果。因此,Function.prototype的原型對象是Object.prototype,其原型函數是Object()。

console.log(Function.prototype.__proto__ === Object.prototype);//true
複製代碼

總結:

【1】函數(Function也是函數)是new Function的結果,因此函數能夠做爲實例對象,其構造函數是Function(),原型對象是Function.prototype。

【2】對象(函數也是對象)是new Object的結果,因此對象能夠做爲實例對象,其構造函數是Object(),原型對象是Object.prototype

【3】Object.prototype的原型對象是null。

相關文章
相關標籤/搜索