詳解JavaScript中基于原型prototype的繼承特性
JavaScript 中的繼承比較奇葩,無法實現(xiàn)接口繼承,只能依靠原型繼承。
原型鏈
原型就是一個對象,通過構(gòu)造函數(shù)創(chuàng)建出來的實例會有指針指向原型得到原型的屬性和方法。這樣,實例對象就帶有構(gòu)造函數(shù)的屬性方法和原型的屬性方法,然后將需要繼承的構(gòu)造函數(shù)的原型指向這個實例,即可擁有這個實例的所有屬性方法實現(xiàn)繼承。
看下面演示代碼:
//聲明超類,通過構(gòu)造函數(shù)和原型添加有關(guān)屬性和方法
function Super(){
this.property = true;
}
Super.prototype.getSuperValue = function() {
return this.property;
};
//聲明子類的構(gòu)造函數(shù)
function SubType() {
this.subproperty = false;
}
//將子類的原型指向超類的實例,得到超類的一切
SubType.prototype = new Super();
SubType.prototype.constructor = SubType;
SubType.prototype.getSubValue = function(){
return this.subproperty;
};
//由子類創(chuàng)建對象,測試是否繼承超類方法和屬性
var instance = new SubType();
console.log(instance.getSuperValue());
所有函數(shù)的默認原型都是 Object 的實例,因此默認原型都會包含一個內(nèi)部指針,指向 Object.prototype。
使用 instanceof 和 isPrototypeOf 可以確定原型和實例的關(guān)系:
instance instanceof Object; Object.prototype.isPrototypeOf(instance);
使用原型鏈的時候,需要謹慎的定義方法。子類需要重寫超類型的某個方法或者擴充,一定要放在替換原型的語句后面,這樣才能生效。此外,通過原型鏈實現(xiàn)繼承時,不能使用對象字面量創(chuàng)建原型方法,這樣會重寫原型鏈:
......
SubType.prototype = new Super();
SubType.prototype = {
....
};
這會更換指針指向新對象,從而重寫了原型鏈。
原型鏈的繼承方法是有缺陷的,主要有兩個問題:
1,來自包含引用類型值的原型,會被所有實例共享。
前面文章介紹過包含引用類型值的原型屬性會被所有實例共享,一個實例修改,其他實例會隨之改變,因此需要在構(gòu)造函數(shù)中定義屬性。而原型鏈繼承的時候,無論超類中屬性是在構(gòu)造函數(shù)還是原型中定義,全部都變成了實例對象被子類繼承,從而對子類的實例產(chǎn)生影響。
2,創(chuàng)建子類型的實例時,不能向超類型的構(gòu)造函數(shù)中傳遞參數(shù)。
原型鏈的繼承,直接將子類原型指向超類的實例,這時候可以向超類傳遞參數(shù)。但是當(dāng)子類創(chuàng)建實例的時候,只能向子類的構(gòu)造函數(shù)傳遞參數(shù),而不能向超類的構(gòu)造函數(shù)傳遞參數(shù)。
因此實際應(yīng)用中,很少單獨使用原型鏈。
相關(guān)的一些代碼實踐
鑒別一個原型屬性
function hasPrototypeProperty(object, name) {
return name in object && !object.hasOwnProperty(name);
}
在構(gòu)造函數(shù)中使用原型對象
function Person(name) {
this.name = name;
}
Person.prototype = {
constructor: Person,
sayName: function () {
console.log(this.name);
},
toString: function() {
}
};
var person1 = new Person('Nicholas');
var person2 = new Person('Greg);
console.log(person1 instanceof Person); // true
console.log(person1.constructor === Person); // true
console.log(person1.constructor === Object); // false
console.log(person2 instanceof Person); // true
console.log(person2.constructor === Person); // true
console.log(person2.constructor === Object); // false
對象繼承
var person1 = {
name: 'Nicholas',
sayName: function () {
console.log(this.name);
}
};
var person2 = Object.create(person1, {
name: {
configurable: true,
enumerable: true,
value: 'Greg',
writable: true
}
});
person1.sayName(); // Nicholas
person2.sayName(); // Greg
console.log(person1.hasOwnProperty('sayName')); // true
console.log(person1.isPropertyOf(person2)); // true
console.log(person2.hasOwnProperty('sayName')); // false
模塊模式
var person = (function () {
var age = 25;
function getAge() {
return age;
}
function growOlder() {
age++;
}
return {
name: 'Nicholas',
getAge: getAge,
growOlder: growOlder
};
}());
作用域的構(gòu)造函數(shù)
function Person(name) {
this.name = name;
}
Person.prototype.sayName = function() {
console.log(this.name);
};
var person1 = Person('Nicholas');
console.log(person1 instanceof Person); // false
console.log(typeof person1); // undefined
console.log(name); // Nicholas
相關(guān)文章
JavaScript字符串對象的concat方法實例(用于連接兩個或多個字符串)
這篇文章主要介紹了JavaScript字符串對象的concat方法實例,這個方法用于連接兩個或多個字符串,平時用+號比較多,所以這個方法可能不太常用,需要的朋友可以參考下2014-10-10
全面了解JavaScript的數(shù)據(jù)類型轉(zhuǎn)換
下面小編就為大家?guī)硪黄媪私釰avaScript的數(shù)據(jù)類型轉(zhuǎn)換。小編覺得挺不錯的,現(xiàn)在就分享給大家看,也給大家做個參考。一起跟隨小編過來看看吧2016-07-07
javascript基礎(chǔ)之?dāng)?shù)據(jù)類型詳解
這篇文章主要介紹了如何理解JavaScript中的數(shù)據(jù)類型,幫助大家更好的學(xué)習(xí)JavaScript,感興趣的朋友可以了解下2021-11-11
探討JavaScript中的Rest參數(shù)和參數(shù)默認值
這篇文章的主要介紹了JavaScript中的Rest參數(shù)和參數(shù)默認值,內(nèi)容很充實,需要了解的朋友可以參考下2015-07-07

