譯者按: 本文簡單的介紹了new, 更多的是介紹原型(prototype),值得一讀。
為了保證可讀性,本文采用意譯而非直譯。另外,本文版權歸原作者所有,翻譯僅用於學習
當你使用new
的時候,會:
- 創建一個新的空對象;
- 將
this
綁定到該對象; - 添加一個名為
__proto__
的新屬性,並且指向構造函數的原型(prototype); - 返回該
this
對象。
如果你沒有特別理解,那么我們接下來用例子來詳細解釋。首先定義一個構造函數Student
,該函數接收兩個參數name
和age
。
function Student(name, age){
this.name = name;
this.age = age;
}
現在我們使用new
來創建一個新的對象:
var first = new Student('John', 26);
到底發生了什么呢?
- 一個新的對象創建,我們叫它
obj
; this
綁定到obj
,任何對this
的引用就是對obj
的引用;__proto__
屬性被添加到obj
對象。obj.__proto__
會指向Student.prototype
;- 該
obj
對象被賦值給first
變量。
我們可以通過打印測試:
console.log(first.name);
// John
console.log(first.age);
// 26
接下來深入看看__proto__
是怎么回事。
原型(Prototype)
每一個JavaScript對象都有一個原型。所有的對象都從它的原型中繼承對象和屬性。
打開瀏覽器開發者控制面板(Windows: Ctrl + Shift + J)(Mac: Cmd + Option + J),輸入之前定義的Student
函數:
function Student(name, age) {
this.name = name;
this.age = age;
}
為了證實每一個對象都有原型,輸入:
Student.prototype;
// Object {...}
你會看到返回了一個對象。現在我們來嘗試定義一個新的對象:
var second = new Student('Jeff', 50);
根據之前的解釋,second
指向的對象會有一個__proto__
屬性,並且應該指向父親的prototype
,我們來測試一下:
second.__proto__ === Student.prototype
// true
Student.prototype.constructor
會指向Student
的構造函數,我們打印出來看看:
Student.prototype.constructor;
// function Student(name, age) {
// this.name = name;
// this.age = age;
// }
好像事情越來越復雜了,我們用圖來形象描述一下:
Student
的構造函數有一個叫.prototype
的屬性,該屬性又有一個.constructor
的屬性反過來指向Student
構造。它們構成了一個環。當我們使用new
去創建一個新的對象,每一個對象都有.__proto__
屬性反過來指向Student.prototype
。
這個設計對於繼承來說很重要。因為原型對象被所有由該構造函數創建的對象共享。當我們添加函數和屬性到原型對象中,其它所有的對象都可以使用。
在本文我們只創建了兩個Student
對象,如果我們創建20,000個,那么將屬性和函數放到prototype
而不是每一個對象將會節省非常很多的存儲和計算資源。
我們來看一個例子:
Student.prototype.sayInfo = function(){
console.log(this.name + ' is ' + this.age + ' years old');
}
我們為Student
的原型添加了一個新的函數sayInfo
-- 所以使用Student
創建的學生對象都可以訪問該函數。
second.sayInfo();
// Jeff is 50 years old
創建一個新的學生對象,再次測試:
var third = new Student('Tracy', 15);
// 如果我們現在打印third, 雖然只會看到年齡和名字這兩個屬性,
// 仍然可以訪問sayInfo函數。
third;
// Student {name: "Tracy", age: 15}
third.sayInfo();
// Tracy is 15 years old
在JavaScript中,首先查看當前對象是否擁有該屬性;如果沒有,看原型中是否有該屬性。這個規則會一直持續,直到成功找到該屬性或則到最頂層全局對象也沒找到而返回失敗。
繼承讓你平時不需要去定義toString()
函數而可以直接使用。因為toString()
這個函數內置在Object
的原型上。每一個我們創建的對象最終都指向Object.prototype
,所以可以調用toString()
。當然, 我們也可以重寫這個函數:
var name = {
toString: function(){
console.log('Not a good idea');
}
};
name.toString();
// Not a good idea
創建的name
對象首先查看是否擁有toString
,如果有就不會去原型中查找。
總結
也許這些概念對你來說有點多,但是當你理解了,使用原型可以讓你寫出更加高效的代碼。
關於Fundebug
Fundebug專注於JavaScript、微信小程序、微信小游戲、支付寶小程序、React Native、Node.js和Java實時BUG監控。 自從2016年雙十一正式上線,Fundebug累計處理了7億+錯誤事件,得到了Google、360、金山軟件、百姓網等眾多知名用戶的認可。歡迎免費試用!
版權聲明
轉載時請注明作者Fundebug以及本文地址:
https://blog.fundebug.com/2017/06/02/javascript-new-for-beginner/