JavaScript進階(四)原型與原型鏈用法實例分析
本文實例講述了JavaScript原型與原型鏈用法。分享給大家供大家參考,具體如下:
一句話說明什么是原型:原型就是一個JavaScript對象,原型能存儲我們的方法,構(gòu)造函數(shù)創(chuàng)建出來的實例對象能夠引用原型中的方法。
一、傳統(tǒng)構(gòu)造函數(shù)的問題
有如下代碼
function Foo(){ this.sayHello = function(){ } }
由于對象是調(diào)用new Foo()
所創(chuàng)建出來的,因此每一個對象在創(chuàng)建的時候,函數(shù) sayHello 都會唄創(chuàng)建一次
那么有沒一個對象都含有一個獨立的,不同的,但是功能邏輯一樣的函數(shù),比如:{} == {}
。
在代碼中方法就會消耗性能,最典型的資源就越是內(nèi)存
這里最好的方法就是將函數(shù)放在構(gòu)造函數(shù)之外,那么在構(gòu)造函數(shù)中引用該函數(shù)即可
function sayHello () {} function Foo () { this.say = sayHello; }
會在開發(fā)中變得困難:引入框架危險,代碼繁冗不好維護。解決方法就是如果外面的函數(shù)不占用其名字,而且在函數(shù)名下。
每一個函數(shù)在定義的時候,有一個神秘對象(就是原型對象,暫且這么稱呼)被創(chuàng)建出來。
每一個由構(gòu)造函數(shù)創(chuàng)建的對象都會默認的連接到該神秘對象上。
var f1 = new Foo(); var f2 = new Foo(); f1.sayHello(); //如果f1沒有sayHello那么就會在Foo.prototype中去找
由構(gòu)造函數(shù)創(chuàng)建出來的眾多對象共享一個對象就是:構(gòu)造函數(shù).prototype
只需要將共享的東西,重復會多占用內(nèi)存的東西放到構(gòu)造函數(shù).prototype
中,那么所有的對象就可以共享了。
function Foo(){} Foo.prototype.sayHello = function(){ console.log("…."); } var f1 = new Foo(); f1.sayHello(); var f2 = new Foo(); f2.sayHello(); console.log(f1.sayHello === f2.sayHello); // true
二、一些相關(guān)概念
類class:在JS中就是構(gòu)造函數(shù)
- 在傳統(tǒng)的面向?qū)ο笳Z言中,使用一個叫類的東西定義模板,然后使用模板創(chuàng)建對象。
- 在構(gòu)造方法中也具有類似的功能,因此也稱其為類
實例(instance)與對象(object)
- 實例一般是指某一個構(gòu)造函數(shù)創(chuàng)建出來的對象,我們稱為XXXX 構(gòu)造函數(shù)的實例
- 實例就是對象。對象是一個泛稱
- 實例與對象是一個近義詞
鍵值對與屬性和方法
- 在JS中鍵值對的集合稱為對象
- 如果值為數(shù)據(jù)(非函數(shù)),就稱該鍵值對為屬性
- 如果值為函數(shù)(方法),就稱該鍵值對為方法method
父類與子類(基類和派生類)
- 傳統(tǒng)的面向?qū)ο笳Z言中使用類來實現(xiàn)繼承那么就有父類、子類的概念
- 父類又稱為基類,子類又稱為派生類
- 在JS中沒有類的概念,在JS中常常稱為父對象,子對象,基對象,派生對象。
三、認識原型
在JavaScript中,原型也是一個對象,通過原型可以實現(xiàn)對象的屬性繼承,JavaScript的對象中都包含了一個[[Prototype]]
內(nèi)部屬性,這個屬性所對應的就是該對象的原型。
[[Prototype]]作為對象的內(nèi)部屬性,是不能被直接訪問的。所以為了方便查看一個對象的原型,F(xiàn)irefox和Chrome中提供了__proto__
這個非標準(不是所有瀏覽器都支持)的訪問器(ECMA引入了標準對象原型訪問器"Object.getPrototype(object)")。
下面通過一個例子來看看原型相關(guān)概念:
function Person() {} // 神秘對象就是Person.prototype //那么只有使用構(gòu)造函數(shù)才可以訪問它 var o = new Person(); //以前不能直接使用o來訪問神秘對象 //現(xiàn)在有了__proto__后, o.__proto__也可以直接訪問神秘對象 //那么o.__proto__ === Person.prototype
神秘對象(原型)中都有一個屬性constructor
,翻譯為 構(gòu)造器 。表示該原型是與什么構(gòu)造函數(shù)聯(lián)系起來的。
__proto__
有什么用?可以訪問原型。由于在開發(fā)中除非特殊要求,不要使用實例去修改原型的成員,因此該屬性開發(fā)時使用較少。但是在調(diào)試過程中非常方便,可以輕易的訪問原型進行查看成員
如果在早期的瀏覽器中使用實例需要訪問原型如何處理?可以使用實例對象訪問構(gòu)造器,然后使用構(gòu)造器訪問原型
var o = new Person(); o.constructor.prototype
如果給實例繼承自原型的屬性賦值
function Foo(); Foo.prototype.name = "test"; var o1 = new Foo(); var o2 = new Foo(); o1.name = "張三"; // 不是修改原型中的name而是自己增加了一個name屬性 console.log(o1.name + ','+ o2.name); // 張三,test
四、構(gòu)造、原型、實例三角結(jié)構(gòu)圖
對于如下代碼:
function Person(){} var p = new Person() console.log(Person.prototype.constructor); //function Person(){} console.log(Person.prototype.constructor.name); //Person console.log(typeof Person.prototype.constructor); //function console.log(p.__prop__); console.log(p.__prop__ === Person.prototype);//true
于是他們的關(guān)系圖如下:
五、對象的原型鏈
凡是對象就有原型,原型也是對象。因此凡是給定一個對象,那么就可以找到他的原型,原型還有原型,那么如此下去,就構(gòu)成一個對象的序列,稱該結(jié)構(gòu)為原型鏈。
問題:
- 原型鏈到底到什么時候是一個頭?
- 一個默認的原型鏈結(jié)構(gòu)是怎樣的?
- 原型鏈結(jié)構(gòu)對已知語法的修正
5.1 原型鏈的結(jié)構(gòu)
凡是使用構(gòu)造函數(shù),創(chuàng)建出對象,并且沒有利用賦值的方式修改原型,就說該對象保留默認的原型鏈。
默認原型鏈結(jié)構(gòu)是什么樣子呢?
function Person(){} var p = new Person(); //p 具有默認的原型鏈
默認的原型鏈結(jié)構(gòu)就是:當前對象 -> 構(gòu)造函數(shù).prototype -> Object.prototype -> null
在實現(xiàn)繼承的時候,有時候會利用替換原型鏈結(jié)構(gòu)的方式實現(xiàn)原型繼承,那么原型鏈結(jié)構(gòu)就會發(fā)送改變
function DunizbCollection(){} DunizbCollection.prototype = []; var arr = new DunizbCollection(); // arr -> [] -> Array.prototype -> Object.prototype -> null
六、函數(shù)的構(gòu)造函數(shù)Function
在JS中使用Function可以實例化函數(shù)對象 。也就是說在JS中函數(shù)與普通對象一樣,也是一個對象類型。函數(shù)是JS中的一等公民。
- 函數(shù)是對象,就可以使用對象的動態(tài)特性
- 函數(shù)是對象,就有構(gòu)造函數(shù)創(chuàng)建函數(shù)
- 函數(shù)是對象,可以創(chuàng)建其它對象
- 函數(shù)是唯一可以限定變量作用域的結(jié)果
要解決的問題
- Function 如何使用
- Function 與函數(shù)的關(guān)系
- 函數(shù)的原型鏈結(jié)構(gòu)
6.1 函數(shù)是Function的實例
語法
new Function( arg0,arg1,arg1,….argN, body );
Function 中的參數(shù)全部是字符串
該構(gòu)造函數(shù)的作用是將參數(shù)鏈接起來組成函數(shù)
- 如果參數(shù)只有一個,那么表示函數(shù)體
- 如果參數(shù)有多個,最后一個參數(shù)表示函數(shù)體,前面的所有參數(shù)表示函數(shù)的參數(shù)
- 如果沒有參數(shù),表示創(chuàng)建一個空函數(shù)
舉例:創(chuàng)建一個打印一句話的函數(shù)
// 傳統(tǒng)的 function foo () { console.log( '你好' ); } //Function var func = new Function( 'console.log( "你好" );' ); // 功能上,這里foo 與 func 等價
再比如,創(chuàng)建一個空函數(shù)
//傳統(tǒng) function foo () {} //Function var func = new Function(); func();
傳入函數(shù)內(nèi)一個數(shù)字,打印該函數(shù)
//傳統(tǒng) function foo ( num ) { console.log( num ); } //Function var func = new Function( "num" ,"console.log( num )" ); func();
6.2 函數(shù)的原型鏈結(jié)構(gòu)
任意的一個函數(shù),都是相當于Function的實例,類似于{}與new Object()的關(guān)系。
function foo () {}
上面的代告訴解析器,有一個對象叫foo,它是一個函數(shù);相當于new Function()
得到一個函數(shù)對象
- 函數(shù)應該有什么屬性?答:
__proto__
- 函數(shù)的構(gòu)造函數(shù)是什么?答:Function
- 函數(shù)應該繼承自Function.prototype
- Function.prototype繼承自O(shè)bject.prototype
對于Function,我們還必須知道
- Object函數(shù)是Function的一個實例
-
Object作為對象是繼承自Function.prototype的,又“Function.prototype”繼承自O(shè)bject.prototype
foo.prototype.__proto__ === Object.prototype // true
- Function是自己的構(gòu)造函數(shù)
- 在JS 中任何對象的老祖宗就是Object.prototype
- 在JS中任何函數(shù)的老祖宗就是Function.prototype
下面繪制出 Function 的構(gòu)造原型實例三角形結(jié)構(gòu)
6.3 為什么要使用Function?
Function是使用字符串構(gòu)建函數(shù),那么就可以在程序運行過程中構(gòu)建函數(shù).
以前的函數(shù)必須一開始就寫好,再經(jīng)過預解析,一步一步的運行
假定從服務器里拿到“[1,2,3,4,5]”,將數(shù)組形式的字符串轉(zhuǎn)換成數(shù)組對象
var arr = ( new Function( 'return ' + str + ' ;' ) )();
感興趣的朋友可以使用在線HTML/CSS/JavaScript代碼運行工具:http://tools.jb51.net/code/HtmlJsRun測試上述代碼運行效果。
更多關(guān)于JavaScript相關(guān)內(nèi)容可查看本站專題:《JavaScript常用函數(shù)技巧匯總》、《javascript面向?qū)ο笕腴T教程》、《JavaScript錯誤與調(diào)試技巧總結(jié)》、《JavaScript數(shù)據(jù)結(jié)構(gòu)與算法技巧總結(jié)》及《JavaScript數(shù)學運算用法總結(jié)》
希望本文所述對大家JavaScript程序設(shè)計有所幫助。
相關(guān)文章
JavaScript 中 avalon綁定屬性總結(jié)
avalon是前端MVVM框架,在js中經(jīng)常會用到。這篇文章主要介紹了JavaScript 中 avalon綁定屬性總結(jié)的相關(guān)資料,需要的朋友可以參考下2016-10-10如何使用JavaScript快速創(chuàng)建一個1到100的數(shù)組
平時寫代碼時,我們會生產(chǎn)一些測試用的數(shù)組數(shù)據(jù),比如[1,100]的數(shù)組值,下面這篇文章主要給大家介紹了關(guān)于如何使用JavaScript快速創(chuàng)建一個1到100數(shù)組的相關(guān)資料,需要的朋友可以參考下2022-08-08JS實現(xiàn)聯(lián)想、自動補齊國家或地區(qū)名稱的功能
這篇文章主要介紹了JS實現(xiàn)聯(lián)想、自動補齊國家或地區(qū)名稱的功能,文中示例代碼非常詳細,幫助大家更好的理解和學習,感興趣的朋友可以了解下2020-07-07前端獲取excel表格數(shù)據(jù)并在瀏覽器展示方法實例
在開發(fā)過程中,難免會碰到用前端來處理excel文件的需求,這篇文章主要給大家介紹了關(guān)于前端獲取excel表格數(shù)據(jù)并在瀏覽器展示方的相關(guān)資料,文中通過代碼介紹的非常詳細,需要的朋友可以參考下2024-08-08