在進入正題前,先了解傳統的面向對象編程(例如Java)中常會涉及到的概念,大體能夠包括:javascript
類:定義對象的特徵。它是對象的屬性和方法的模板定義。java
對象(或稱實例):類的一個實例。編程
屬性:對象的特徵,好比顏色、尺寸等。瀏覽器
方法:對象的行爲,好比行走、說話等。框架
構造函數:對象初始化的瞬間被調用的方法。函數
繼承:子類能夠繼承父類的特徵。例如,貓繼承了動物的通常特性。工具
封裝:一種把數據和相關的方法綁定在一塊兒使用的方法。性能
抽象:結合複雜的繼承、方法、屬性的對象可以模擬現實的模型。this
多態:不一樣的類能夠定義相同的方法或屬性。spa
在JavaScript的面向對象編程中大致也包括這些。不過在稱呼上可能稍有不一樣,例如,JavaScript中沒有原生的「類」的概念, 而只有對象的概念。所以,隨着你認識的深刻,咱們會混用對象、實例、構造函數等概念。
在JavaScript中,咱們一般可使用構造函數來建立特定類型的對象。諸如Object和Array這樣的原生構造函數,在運行時會自動出如今執行環境中。 此外,咱們也能夠建立自定義的構造函數。例如:
function Person(name, age, job) { this.name = name; this.age = age; this.job = job;}var person1 = new Person('Weiwei', 27, 'Student');var person2 = new Person('Lily', 25, 'Doctor');
按照慣例,構造函數始終都應該以一個大寫字母開頭(和Java中定義的類同樣),普通函數則小寫字母開頭。 要建立Person
的新實例,必須使用new
操做符。以這種方式調用構造函數實際上會經歷如下4個步驟:
建立一個新對象(實例)
將構造函數的做用域賦給新對象(也就是重設了this
的指向,this
就指向了這個新對象)
執行構造函數中的代碼(爲這個新對象添加屬性)
返回新對象
有關new
操做符的更多內容請參考這篇文檔。
在上面的例子中,咱們建立了Person
的兩個實例person1
和person2
。 這兩個對象默認都有一個constructor
屬性,該屬性指向它們的構造函數Person
,也就是說:
console.log(person1.constructor == Person); //trueconsole.log(person2.constructor == Person); //true
咱們可使用instanceof
操做符進行類型檢測。咱們建立的全部對象既是Object
的實例,同時也是Person
的實例。 由於全部的對象都繼承自Object
。
console.log(person1 instanceof Object); //trueconsole.log(person1 instanceof Person); //trueconsole.log(person2 instanceof Object); //trueconsole.log(person2 instanceof Person); //true
咱們不建議在構造函數中直接定義方法,若是這樣作的話,每一個方法都要在每一個實例上從新建立一遍,這將很是損耗性能。 ——不要忘了,ECMAScript中的函數是對象,每定義一個函數,也就實例化了一個對象。
幸運的是,在ECMAScript中,咱們能夠藉助原型對象來解決這個問題。
咱們建立的每一個函數都有一個prototype
屬性,這個屬性是一個指針,指向該函數的原型對象, 該對象包含了由特定類型的全部實例共享的屬性和方法。也就是說,咱們能夠利用原型對象來讓全部對象實例共享它所包含的屬性和方法。
function Person(name, age, job) { this.name = name; this.age = age; this.job = job;}// 經過原型模式來添加全部實例共享的方法// sayName() 方法將會被Person的全部實例共享,而避免了重複建立Person.prototype.sayName = function () { console.log(this.name);};var person1 = new Person('Weiwei', 27, 'Student');var person2 = new Person('Lily', 25, 'Doctor');console.log(person1.sayName === person2.sayName); // trueperson1.sayName(); // Weiweiperson2.sayName(); // Lily
正如上面的代碼所示,經過原型模式定義的方法sayName()
爲全部的實例所共享。也就是, person1
和person2
訪問的是同一個sayName()
函數。一樣的,公共屬性也可使用原型模式進行定義。例如:
function Chinese (name) { this.name = name;}Chinese.prototype.country = 'China'; // 公共屬性,全部實例共享
如今咱們來深刻的理解一下什麼是原型對象。
只要建立了一個新函數,就會根據一組特定的規則爲該函數建立一個prototype
屬性,這個屬性指向函數的原型對象。 在默認狀況下,全部原型對象都會自動得到一個constructor
屬性,這個屬性包含一個指向prototype
屬性所在函數的指針。 也就是說:Person.prototype.constructor
指向Person
構造函數。
建立了自定義的構造函數以後,其原型對象默認只會取得constructor
屬性;至於其餘方法,則都是從Object
繼承而來的。 當調用構造函數建立一個新實例後,該實例內部將包含一個指針(內部屬性),指向構造函數的原型對象。ES5中稱這個指針爲[[Prototype]]
, 在Firefox、Safari和Chrome在每一個對象上都支持一個屬性__proto__
(目前已被廢棄);而在其餘實現中,這個屬性對腳本則是徹底不可見的。 要注意,這個連接存在於實例與構造函數的原型對象之間,而不是實例與構造函數之間。
這三者關係的示意圖以下:
上圖展現了Person
構造函數、Person
的原型對象以及Person
現有的兩個實例之間的關係。
Person.prototype
指向了原型對象
Person.prototype.constructor
又指回了Person
構造函數
Person
的每一個實例person1
和person2
都包含一個內部屬性(一般爲__proto__
),person1.__proto__
和person2.__proto__
指向了原型對象
從上圖咱們發現,雖然Person
的兩個實例都不包含屬性和方法,但咱們卻能夠調用person1.sayName()
。 這是經過查找對象屬性的過程來實現的。
搜索首先從對象實例自己開始(實例person1
有sayName
屬性嗎?——沒有)
若是沒找到,則繼續搜索指針指向的原型對象(person1.__proto__
有sayName
屬性嗎?——有)
這也是多個對象實例共享原型所保存的屬性和方法的基本原理。
注意,若是咱們在對象的實例中重寫了某個原型中已存在的屬性,則該實例屬性會屏蔽原型中的那個屬性。 此時,可使用delete
操做符刪除實例上的屬性。
Object.getPrototypeOf()
根據ECMAScript標準,someObject.[[Prototype]]
符號是用於指派 someObject
的原型。 這個等同於 JavaScript 的 __proto__
屬性(現已棄用)。 從ECMAScript 5開始, [[Prototype]]
能夠用Object.getPrototypeOf()
和Object.setPrototypeOf()
訪問器來訪問。
其中Object.getPrototypeOf()
在全部支持的實現中,這個方法返回[[Prototype]]
的值。例如:
person1.__proto__ === Object.getPrototypeOf(person1); // true Object.getPrototypeOf(person1) === Person.prototype; // true
也就是說,Object.getPrototypeOf(p1)
返回的對象實際就是這個對象的原型。 這個方法的兼容性請參考該連接。
Object.keys()
要取得對象上全部可枚舉的實例屬性,可使用ES5中的Object.keys()
方法。例如:
Object.keys(p1); // ["name", "age", "job"]
此外,若是你想要獲得全部實例屬性,不管它是否可枚舉,均可以使用Object.getOwnPropertyName()
方法。
在上面的代碼中,若是咱們要添加原型屬性和方法,就要重複的敲一遍Person.prototype
。爲了減小這個重複的過程, 更常見的作法是用一個包含全部屬性和方法的對象字面量來重寫整個原型對象。 參考資料。
function Person(name, age, job) { this.name = name; this.age = age; this.job = job;}Person.prototype = { // 這裏務必要從新將構造函數指回Person構造函數,不然會指向這個新建立的對象 constructor: Person, // Attention! sayName: function () { console.log(this.name); }};var person1 = new Person('Weiwei', 27, 'Student');var person2 = new Person('Lily', 25, 'Doctor');console.log(person1.sayName === person2.sayName); // trueperson1.sayName(); // Weiweiperson2.sayName(); // Lily
在上面的代碼中特地包含了一個constructor
屬性,並將它的值設置爲Person
,從而確保了經過該屬性可以訪問到適當的值。 注意,以這種方式重設constructor
屬性會致使它的[[Enumerable]]
特性設置爲true
。默認狀況下,原生的constructor
屬性是不可枚舉的。 你可使用Object.defineProperty()
:
// 重設構造函數,只適用於ES5兼容的瀏覽器Object.defineProperty(Person.prototype, "constructor", { enumerable: false, value: Person});
建立自定義類型的最多見方式,就是組合使用構造函數模式與原型模式。構造函數模式用於定義實例屬性, 而原型模式用於定義方法和共享的屬性。結果,每一個實例都會有本身的一份實例屬性的副本,但同時又共享着對方的引用, 最大限度的節省了內存。
大多的面嚮對象語言都支持兩種繼承方式:接口繼承和實現繼承。ECMAScript只支持實現繼承,並且其實現繼承主要依靠原型鏈來實現。
使用原型鏈做爲實現繼承的基本思想是:利用原型讓一個引用類型繼承另外一個引用類型的屬性和方法。首先咱們先回顧一些基本概念:
每一個構造函數都有一個原型對象(prototype
)
原型對象包含一個指向構造函數的指針(constructor
)
實例都包含一個指向原型對象的內部指針([[Prototype]]
)
若是咱們讓原型對象等於另外一個類型的實現,結果會怎麼樣?顯然,此時的原型對象將包含一個指向另外一個原型的指針, 相應的,另外一個原型中也包含着一個指向另外一個構造函數的指針。假如另外一個原型又是另外一個類型的實例,那麼上述關係依然成立, 如此層層遞進,就構成了實例與原型的鏈條。 更詳細的內容能夠參考這個連接。 先看一個簡單的例子,它演示了使用原型鏈實現繼承的基本框架:
function Father () { this.fatherValue = true;}Father.prototype.getFatherValue = function () { console.log(this.fatherValue);};function Child () { this.childValue = false;}// 實現繼承:繼承自FatherChild.prototype = new Father();Child.prototype.getChildValue = function () { console.log(this.childValue);};var instance = new Child();instance.getFatherValue(); // trueinstance.getChildValue(); // false
在上面的代碼中,原型鏈繼承的核心語句是Child.prototype = new Father()
,它實現了Child
對Father
的繼承, 而繼承是經過建立Father
的實例,並將該實例賦給Child.prototype
實現的。
實現的本質是重寫原型對象,代之以一個新類型的實例。也就是說,原來存在於Father
的實例中的全部屬性和方法, 如今也存在於Child.prototype
中了。
這個例子中的實例以及構造函數和原型之間的關係以下圖所示:
在上面的代碼中,咱們沒有使用Child
默認提供的原型,而是給它換了一個新原型;這個新原型就是Father
的實例。 因而,新原型不只具備了做爲一個Father
的實例所擁有的所有屬性和方法。並且其內部還有一個指針[[Prototype]]
,指向了Father
的原型。
instance
指向Child
的原型對象
Child
的原型對象指向Father
的原型對象
getFatherValue()
方法仍然還在Father.prototype
中
可是,fatherValue
則位於Child.prototype
中
instance.constructor
如今指向的是Father
由於fatherValue
是一個實例屬性,而getFatherValue()
則是一個原型方法。既然Child.prototype
如今是Father
的實例, 那麼fatherValue
固然就位於該實例中。
經過實現原型鏈,本質上擴展了本章前面介紹的原型搜索機制。例如,instance.getFatherValue()
會經歷三個搜索步驟:
搜索實例
搜索Child.prototype
搜索Father.prototype
Object
全部的函數都默認原型都是Object
的實例,所以默認原型都會包含一個內部指針[[Prototype]]
,指向Object.prototype
。 這也正是全部自定義類型都會繼承toString()
、valueOf()
等默認方法的根本緣由。因此, 咱們說上面例子展現的原型鏈中還應該包括另一個繼承層次。關於Object
的更多內容,能夠參考這篇博客。
也就是說,Child
繼承了Father
,而Father
繼承了Object
。當調用了instance.toString()
時, 實際上調用的是保存在Object.prototype
中的那個方法。
首先是順序,必定要先繼承父類,而後爲子類添加新方法。
其次,使用原型鏈實現繼承時,不能使用對象字面量建立原型方法。由於這樣作就會重寫原型鏈,以下面的例子所示:
function Father () { this.fatherValue = true;}Father.prototype.getFatherValue = function () { console.log(this.fatherValue);};function Child () { this.childValue = false;}// 繼承了Father// 此時的原型鏈爲 Child -> Father -> ObjectChild.prototype = new Father();// 使用字面量添加新方法,會致使上一行代碼無效// 此時咱們設想的原型鏈被切斷,而是變成 Child -> ObjectChild.prototype = { getChildValue: function () { console.log(this.childValue); }};var instance = new Child();instance.getChildValue(); // falseinstance.getFatherValue(); // error!
在上面的代碼中,咱們連續兩次修改了Child.prototype
的值。因爲如今的原型包含的是一個Object
的實例, 而非Father
的實例,所以咱們設想中的原型鏈已經被切斷——Child
和Father
之間已經沒有關係了。
最後,在建立子類型的實例時,不能向超類型的構造函數中傳遞參數。實際上,應該說是沒有辦法在不影響全部對象實例的狀況下, 給超類型的構造函數傳遞參數。所以,咱們不多單獨使用原型鏈。
借用構造函數(constructor stealing)的基本思想以下:即在子類構造函數的內部調用超類型構造函數。
function Father (name) { this.name = name; this.colors = ['red', 'blue', 'green'];}function Child (name) { // 繼承了Father,同時傳遞了參數 Father.call(this, name);}var instance1 = new Child("weiwei");instance1.colors.push('black');console.log(instance1.colors); // [ 'red', 'blue', 'green', 'black' ]console.log(instance1.name); // weiweivar instance2 = new Child("lily");console.log(instance2.colors); // [ 'red', 'blue', 'green' ]console.log(instance2.name); // lily
爲了確保Father
構造函數不會重寫子類型的屬性,能夠在調用超類型構造函數後,再添加應該在子類型中定義的屬性。
同構造函數同樣,沒法實現方法的複用。
一般,咱們會組合使用原型鏈繼承和借用構造函數來實現繼承。也就是說,使用原型鏈實現對原型屬性和方法的繼承, 而經過借用構造函數來實現對實例屬性的繼承。這樣,既經過在原型上定義方法實現了函數複用,又可以保證每一個實例都有它本身的屬性。 咱們改造最初的例子以下:
// 父類構造函數function Person (name, age, job) { this.name = name; this.age = age; this.job = job;}// 父類方法Person.prototype.sayName = function () { console.log(this.name);};// --------------// 子類構造函數function Student (name, age, job, school) { // 繼承父類的全部實例屬性 Person.call(this, name, age, job); this.school = school; // 添加新的子類屬性}// 繼承父類的原型方法Student.prototype = new Person();// 新增的子類方法Student.prototype.saySchool = function () { console.log(this.school);};var person1 = new Person('Weiwei', 27, 'Student');var student1 = new Student('Lily', 25, 'Doctor', "Southeast University");console.log(person1.sayName === student1.sayName); // trueperson1.sayName(); // Weiweistudent1.sayName(); // Lilystudent1.saySchool(); // Southeast University
組合集成避免了原型鏈和借用構造函數的缺陷,融合了它們的優勢,成爲了JavaScript中最經常使用的繼承模式。 並且,instanceof
和isPropertyOf()
也可以用於識別基於組合繼承建立的對象。
Object.create()
在上面,咱們繼承父類的原型方法使用的是Student.prototype = new Person()
。 這樣作有不少的問題。 改進方法是使用ES5中新增的Object.create()
。能夠調用這個方法來建立一個新對象。新對象的原型就是調用create()
方法傳入的第一個參數:
Student.prototype = Object.create(Person.prototype);console.log(Student.prototype.constructor); // [Function: Person]// 設置 constructor 屬性指向 StudentStudent.prototype.constructor = Student;
詳細用法能夠參考文檔。 關於Object.create()
的實現,咱們能夠參考一個簡單的polyfill:
function createObject(proto) { function F() { } F.prototype = proto; return new F();}// Usage:Student.prototype = createObject(Person.prototype);
從本質上講,createObject()
對傳入其中的對象執行了一次淺複製。
ES6中引入了一套新的關鍵字用來實現class。 JavaScript仍然是基於原型的,這些新的關鍵字包括class、 constructor、 static、 extends、 和super。
對前面的代碼修改以下:
'use strict';class Person { constructor (name, age, job) { this.name = name; this.age = age; this.job = job; } sayName () { console.log(this.name); }}class Student extends Person { constructor (name, age, school) { super(name, age, 'Student'); this.school = school; } saySchool () { console.log(this.school); }}var stu1 = new Student('weiwei', 20, 'Southeast University');var stu2 = new Student('lily', 22, 'Nanjing University');stu1.sayName(); // weiweistu1.saySchool(); // Southeast Universitystu2.sayName(); // lilystu2.saySchool(); // Nanjing University
class
是JavaScript中現有基於原型的繼承的語法糖。ES6中的類並非一種新的建立對象的方法,只不過是一種「特殊的函數」, 所以也包括類表達式和類聲明, 但須要注意的是,與函數聲明不一樣的是,類聲明不會被提高。 參考連接
constructor
constructor()
方法是有一種特殊的和class
一塊兒用於建立和初始化對象的方法。注意,在ES6類中只能有一個名稱爲constructor
的方法, 不然會報錯。在constructor()
方法中能夠調用super
關鍵字調用父類構造器。若是你沒有指定一個構造器方法, 類會自動使用一個默認的構造器。參考連接
static
靜態方法就是能夠直接使用類名調用的方法,而無需對類進行實例化,固然實例化後的類也沒法調用靜態方法。 靜態方法常被用於建立應用的工具函數。參考連接
extends
extends
關鍵字能夠用於繼承父類。使用extends
能夠擴展一個內置的對象(如Date
),也能夠是自定義對象,或者是null
。
super
super
關鍵字用於調用父對象上的函數。 super.prop
和super[expr]
表達式在類和對象字面量中的任何方法定義中都有效。
super([arguments]); // 調用父類構造器super.functionOnParent([arguments]); // 調用父類中的方法
若是是在類的構造器中,須要在this
關鍵字以前使用。參考連接
本文對JavaScript的面向對象機制進行了較爲深刻的解讀,尤爲是構造函數和原型鏈方式實現對象的建立、繼承、以及實例化。 此外,本文還簡要介紹瞭如在ES6中編寫面向對象代碼。