由淺入深講解Javascript繼承機制與simple-inheritance源碼分析
老生常談的問題,大部分人也不一定可以系統(tǒng)的理解。Javascript語言對繼承實現(xiàn)的并不好,需要工程師自己去實現(xiàn)一套完整的繼承機制。下面我們由淺入深的系統(tǒng)掌握使用javascript繼承的技巧。
1. 直接使用原型鏈
這是最簡粗暴的一種方式,基本沒法用于具體的項目中。一個簡單的demo如下:
function SuperType(){ this.property = true; } SuperType.prototype.getSuperValue = function(){ return this.property; } function SubType(){ this.subproperty = false; } //繼承 SubType.prototype = new SuperType(); SubType.prototype.getSubValue = function(){ return this.subproperty; } var instance = new SubType();
這種方式的問題是原型中的屬性會被所用實例共享,通過一個實例改變一個繼承過來的屬性時,會影響到其他實例。,這顯然不是一種常規(guī)意義上的繼承。
2.使用構造函數(shù)
構造函數(shù)本質上也只是一個函數(shù)而已,可以在任何作用域中調用,在子構造函數(shù)中調用父構造函數(shù),就可以實現(xiàn)簡單的繼承。
function SuperType(){ this.colors = {"red","blue","green"} } function SubType(){ SuperType.call(this); } var instance = new SubType();
這種實現(xiàn)避免了多個實例共享屬性的問題,但是又出現(xiàn)了新的問題,比如沒法共享函數(shù),而且 instance instanceof SuperType 為false。
3. 組合使用原型和構造函數(shù)
function SuperType(name){ this.name = name; this.colors = {"red","blue","green"} } SuperType.prototype.sayName = function(){ //code } function SubType(name,age){ SuperType.call(this,name); this.age = age; } SubType.prototype = new SuperType(); var instance = new SubType();
組合使用原型和構造函數(shù)是javascript中最常用的繼承模式。使用這種方式,每個實例都有自己的屬性,同時可以共享原型中的方法。但是這種方式的缺點是:無論什么情況,都會調用兩次超類構造函數(shù)。一次是在創(chuàng)建子類原型時,另一次是在子類構造函數(shù)內部。這種問題該怎么解決呢?
4. 寄生組合式繼承
SubType的原型并不一定非要是SuperType的實例,只需是一個構造函數(shù)的原型是SuperType的原型的普通對象就可以了。Douglas Crockford的方法如下:
function obejct(o){ function F(){}; F.prototype = o; return new F(); }
其實這也就是ES5中Object.create的實現(xiàn)。那么我們可以修改本文中的第3種方案:
function inheritPrototype(subType,superType){ var prototype = object(superType.prototype); prototype.constructor = subType; subType.prototype = prototype; } function SuperType(name){ this.name = name; this.colors = {"red","blue","green"} } SuperType.prototype.sayName = function(){ //code } function SubType(name,age){ SuperType.call(this,name); this.age = age; } inheritPrototype(SubType,SuperType); var instance = new SubTYpe();
其實寄生組合式繼承已經是一種非常好的繼承實現(xiàn)機制了,足以應付日常使用。如果我們提出更高的要求:比如如何在子類中調用父類的方法呢?
5.simple-inheritance庫的實現(xiàn)
看這么難懂的代碼,起初我是拒絕的,但是深入之后才發(fā)現(xiàn)大牛就是大牛,精妙思想無處不在。我對每一行代碼都有詳細的注釋。如果你想了解細節(jié),請務必詳細研究,讀懂每一行。我覺得這個實現(xiàn)最精妙的地方就是按需重寫父類方法,在實例對象中可以通過_super調用父類的同名方法,類似于java的實現(xiàn)。
(function(){ //initializing用于控制類的初始化,非常巧妙,請留意下文中使用技巧 //fnTest返回一個正則比表達式,用于檢測函數(shù)中是否含有_super,這樣就可以按需重寫,提高效率。當然瀏覽器如果不支持的話就返回一個通用正則表達式 var initializing = false,fnTest = /xyz/.test(function(){xyz;}) ? /\b_super\b/ : /.*/; //所有類的基類Class,這里的this一般是window對象 this.Class = function(){}; //對基類添加extend方法,用于從基類繼承 Class.extend = function(prop){ //保存當前類的原型 var _super = this.prototype; //創(chuàng)建當前類的對象,用于賦值給子類的prototype,這里非常巧妙的使用父類實例作為子類的原型,而且避免了父類的初始化(通過閉包作用域的initializing控制) initializing = true; var prototype = new this(); initializing = false; //將參數(shù)prop中賦值到prototype中,這里的prop中一般是包括init函數(shù)和其他函數(shù)的對象 for(var name in prop){ //對應重名函數(shù),需要特殊處理,處理后可以在子函數(shù)中使用this._super()調用父類同名構造函數(shù), 這里的fnTest很巧妙:只有子類中含有_super字樣時才處理從寫以提高效率 prototype[name] = typeof prop[name] == "function" && typeof _super[name] == "function" && fnTest.test(prop[name])? (function(name,fn){ return function(){ //_super在這里是我們的關鍵字,需要暫時存儲一下 var tmp = this._super; //這里就可以通過this._super調用父類的構造函數(shù)了 this._super = _super[name]; //調用子類函數(shù) fn.apply(this,arguments); //復原_super,如果tmp為空就不需要復原了 tmp && (this._super = tmp); } })(name,prop[name]) : prop[name]; } //當new一個對象時,實際上是調用該類原型上的init方法,注意通過new調用時傳遞的參數(shù)必須和init函數(shù)的參數(shù)一一對應 function Class(){ if(!initializing && this.init){ this.init.apply(this,arguments); } } //給子類設置原型 Class.prototype = prototype; //給子類設置構造函數(shù) Class.prototype.constructor = Class; //設置子類的extend方法,使得子類也可以通過extend方法被繼承 Class.extend = arguments.callee; return Class; } })();
通過使用simple-inheritance庫,我們就可以通過很簡單的方式實現(xiàn)繼承了,是不是發(fā)現(xiàn)特別像強類型語言的繼承。
var Human = Class.extend({ init: function(age,name){ this.age = age; this.name = name; }, say: function(){ console.log("I am a human"); } }); var Man = Human.extend({ init: function(age,name,height){ this._super(age,name); this.height = height; }, say: function(){ this._super(); console.log("I am a man"); } }); var man = new Man(21,'bob','191'); man.say();
由淺入深講解Javascript繼承機制與simple-inheritance源碼分析,希望本文分享能夠幫助到大家。
相關文章
javascript使用smipleChart實現(xiàn)簡單圖表
這篇文章主要介紹了javascript使用smipleChart實現(xiàn)簡單圖表的方法及示例分享,需要的朋友可以參考下2015-01-01jscript之List Excel Color Values
jscript之List Excel Color Values...2007-06-06用Javascript 和 CSS 實現(xiàn)腳注(Footnote)效果
腳注(Footnote)是向用戶提供更多信息的一個最佳途徑,也是主體信息的一個有效補充,常見于各種印刷書籍中。2009-09-09Javascript aop(面向切面編程)之around(環(huán)繞)分析
這篇文章主要介紹了Javascript aop(面向切面編程)之around(環(huán)繞) ,需要的朋友可以參考下2015-05-05