JavaScript中的幾種繼承方法示例
1.原型鏈繼承
原理: 子類原型指向父類實例對象實現(xiàn)原型共享,即Son.prototype = new Father()。
這里先簡單介紹下原型
js中每個對象都有一個__proto__屬性,這個屬性指向的就是該對象的原型。js中每個函數(shù)都有一個prototype屬性,這個屬性指向該函數(shù)作為構(gòu)造函數(shù)調(diào)用時創(chuàng)建的實例的原型。原型對象上有一個constructor屬性,指向創(chuàng)建該對象的構(gòu)造函數(shù),該屬性不可枚舉。
var obj = {};
obj.__proto__ === Object.prototype; //true
console.log(Object.prototype.constructor) // ƒ Object
當(dāng)我們訪問一個對象的屬性或者方法時,如果找不到,則會通過原型向上尋找,若原型上也未找到,則會去原型的原型上面去找。比如我要調(diào)用obj.toString方法時,在自身并未找到toString方法,則會去原型上去尋找,即在Object.prototype上去尋找,找到后運行該方法。
var obj = {};
obj.toString();
obj.__proto__.toString(); //obj.__proto__和Object.prototype指向的是一個對象,自然就能訪問Object.prototype上的toString方法啦
注意:原型鏈的終點是null,使用bind方法返回的函數(shù)沒有prototype屬性。
var obj = {};
function fn(){};
fn.bind(obj).prototype; // undefined
Object.prototype.__proto__; // null
原型鏈接繼承
function Father(age){
this.age = age;
this.color = ['red','pink']
}
Father.prototype.sayHello = function(){
console.log('hello')
}
function Son(sex){
this.sex = sex
}
console.log(Son.prototype.constructor) // ƒ Son
Son.prototype = new Father(15) // 原型鏈繼承關(guān)鍵
var son = new Son('男')
son.color.push('black')
var son2 = new Son('女')
son.sayHello() // hello
son.sayHello === son2.sayHello //true
console.log(son2.color) // ['red','pink','black']
console.log(Son.prototype.constructor) // ƒ Father
可以看到通過原型鏈實現(xiàn)繼承,原型上引用類型的值會被所有實例共享。子類的constructor指向會發(fā)生改變,而且在創(chuàng)建子類實例時不可以向父類構(gòu)造函數(shù)傳遞參數(shù)??梢允謩影炎宇恈onstructor屬性指回其構(gòu)造函數(shù)。
//寫法一
Son.prototype.constructor = Son // 這種寫法有點缺點,它會讓constructor屬性變的可以枚舉。
//寫法二
Object.defineProperty(Son.prototype,'constructor',{
enumerable:false, // 設(shè)置不可枚舉
value:Son
})
2.構(gòu)造函數(shù)繼承
原理:在子類構(gòu)造函數(shù)中通過apply或者call調(diào)用父類構(gòu)造函數(shù)來繼承屬性或方法。
function Father(name){
this.color = ['red']
this.sayHello = function(){
console.log('hello')
}
}
Father.prototype.sayName = function(){
console.log('zs')
}
function Son(num,name){
Father.call(this,name) //實現(xiàn)繼承的關(guān)鍵代碼
this.num = num
}
var son = new Son(10,'zs')
var son2 = new Son(15,'ls')
son.color.push('pink')
console.log(son2.color) // ['red']
son.sayName() //報錯 son.sayName is not a function
console.log(son.sayHello === son2.sayHello) //false
可以看出通過構(gòu)造函數(shù)實現(xiàn)繼承,解決了原型鏈繼承不能向父類傳參以及引用類型值共享的問題。但這種繼承方法卻不能訪問父類構(gòu)造函數(shù)原型上的方法和屬性,而且定義在父類構(gòu)造函數(shù)中的方法也不能復(fù)用。
3.組合式繼承
組合繼承,有時候也叫偽經(jīng)典繼承,它是將原型鏈繼承和構(gòu)造函數(shù)繼承結(jié)合到一起的一種繼承模式。實現(xiàn)思路是通過原型鏈實現(xiàn)對原型屬性和方法的繼承,通過借用構(gòu)造函數(shù)實現(xiàn)對實例屬性的繼承。
function Father(name){
this.color = ['red']
}
Father.prototype.sayName = function(){
console.log('zs')
}
function Son(num,name){
Father.call(this,name) //繼承實例屬性
this.num = num
}
Son.prototype = new Father() //繼承原型上屬性
Son.prototype.constructor = Son
var son = new Son(10,'zs')
var son2 = new Son(15,'ls')
son.color.push('pink')
console.log(son.color,son2.color) //['red','pink'] ['red']
son.sayName() // zs
組合式繼承避免了原型鏈繼承和構(gòu)造函數(shù)繼承的缺點,融合了它們的優(yōu)點,成為JavaScript中常用的一種繼承模式。
4.寄生式繼承
寄生式繼承與工廠模式類似,一般用來繼承對象。即創(chuàng)建一個封裝繼承的函數(shù),在函數(shù)內(nèi)部復(fù)制一份該對象,對復(fù)制的對象進行處理,返回復(fù)制的對象。
function createAnother(obj){
var clone = Object.create(obj)
clone.name = 'zs'
clone.sayHello = function(){
console.log('hello')
}
return clone
}
var obj = {age:15}
var newObj = createAnother(obj) // 15
console.log(newObj.name) // zs
newObj.sayHello() // hello
5.寄生組合式繼承
前面說到過組合式繼承是Javascript中最常用的繼承模式,不過這種模式也有自己的不足,它會調(diào)用兩次父類構(gòu)造函數(shù)。第一次是在將子類原型指向父類實例的時候,第二次是在子類構(gòu)造函數(shù)中調(diào)用的。
function Father(name){
this.name = name
}
function Son(num,name){
Father.call(this,name) // 第二次調(diào)用
}
Son.prototype = new Father('ls') // 第一次調(diào)用
var son = new Son(10,'zs')
console.log(son)
在第一次調(diào)用的時候,Son.prototype會繼承name這個屬性,第二次調(diào)用時,實例對象會繼承name。當(dāng)我們獲取實例對象的name屬性時因為實例對象上有該屬性,所以是不會去原型上去尋找的,相當(dāng)于實例對象上的name屬性把原型上的name屬性給屏蔽掉了,所以原型上的這個屬性是多余的。

為了解決這個問題,就有了寄生組合式繼承。主要思路就是創(chuàng)建一個函數(shù)完成原型鏈繼承和constructor的指向問題,然后通過構(gòu)造函數(shù)繼承屬性。
// 復(fù)制一個父類的原型指向,將子類的原型指向復(fù)制的父類原型,達到不用調(diào)用父類構(gòu)造函數(shù)就能繼承其原型上的方法的效果。
function inherit(Sup,Sub){
var prototype = Object.create(Sup.prototype)
Sub.prototype = prototype
prototype.constructor = Sub
}
function Father(name){
this.name = name
}
function Son(name){
Father.call(this,name)
}
inherit(Father,Son)
var son = new Son('zs')
console.log(son)

以上就是JavaScript中常用的幾種繼承方式啦。
到此這篇關(guān)于JavaScript中的幾種繼承方法的文章就介紹到這了,更多相關(guān)JavaScript繼承方法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
JavaScript實現(xiàn)的原生態(tài)Tab標(biāo)簽頁功能【兼容IE6】
這篇文章主要介紹了JavaScript實現(xiàn)的原生態(tài)Tab標(biāo)簽頁功能,可兼容IE6及谷歌等瀏覽器,涉及javascript事件響應(yīng)及頁面元素動態(tài)操作相關(guān)實現(xiàn)技巧,需要的朋友可以參考下2017-09-09

