首頁  >  文章  >  web前端  >  由淺入深講解Javascript繼承機制與simple-inheritance源碼分析_javascript技巧

由淺入深講解Javascript繼承機制與simple-inheritance源碼分析_javascript技巧

WBOY
WBOY原創
2016-05-16 15:25:511221瀏覽

老生常談的問題,大部分人也不一定可以系統化的理解。 Javascript語言對繼承實作的並不好,需要工程師自己去實作一套完整的繼承機制。下面我們由淺入深的系統掌握使用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();

這種方式的問題是原型中的屬性會被所用實例共享,透過一個實例改變一個繼承過來的屬性時,會影響到其他實例。 ,這顯然不是一種常規意義上的繼承。

2.使用建構子

構造函數本質上也只是一個函數而已,可以在任何作用域中調用,在子構造函數中調用父構造函數,就可以實現簡單的繼承。

function SuperType(){
  this.colors = {"red","blue","green"}
}
function SubType(){
  SuperType.call(this);  
}
var instance = new SubType();

這種實作避免了多個實例共享屬性的問題,但是又出現了新的問題,例如沒法共享函數,而且 instance instanceof SuperType 為false。

3. 組合使用原型與建構子

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();

組合使用原型和建構子是javascript中最常用的繼承模式。使用這種方式,每個實例都有自己的屬性,同時可以共享原型中的方法。但是這種方式的缺點是:無論是什麼情況,都會呼叫兩次超類別建構子。一次是在創建子類別原型時,另一次是在子類別建構子內部。這種問題該怎麼解決呢?

4. 寄生組合式繼承

SubType的原型並不一定非要是SuperType的實例,只需是一個建構子的原型是SuperType的原型的普通物件就可以了。 Douglas Crockford的方法如下:

function obejct(o){
  function F(){};
  F.prototype = o;
  return new F();
}

其實這也就是ES5中Object.create的實作。那我們可以修改本文中的第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();

其實寄生組合式繼承已經是一種非常好的繼承實作機制了,足以應付日常使用。如果我們提出更高的要求:例如如何在子類別中呼叫父類別的方法呢?

5.simple-inheritance函式庫的實作

看這麼難懂的程式碼,起初我是拒絕的,但是深入之後才發現大牛就是大牛,精妙思想無所不在。我對每一行程式碼都有詳細的註解。如果你想了解細節,請務必詳細研究,讀懂每一行。我覺得這個實作最精妙的地方就是按需重寫父類別方法,在實例物件中可以透過_super呼叫父類別的同名方法,類似java的實作。

(function(){
  //initializing用于控制类的初始化,非常巧妙,请留意下文中使用技巧
  //fnTest返回一个正则比表达式,用于检测函数中是否含有_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;
    //创建当前类的对象,用于赋值给子类的prototype,这里非常巧妙的使用父类实例作为子类的原型,而且避免了父类的初始化(通过闭包作用域的initializing控制)
    initializing = true;
    var prototype = new this();   
    initializing = false;
    //将参数prop中赋值到prototype中,这里的prop中一般是包括init函数和其他函数的对象
    for(var name in prop){
      //对应重名函数,需要特殊处理,处理后可以在子函数中使用this._super()调用父类同名构造函数, 这里的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调用父类的构造函数了       
          this._super = _super[name];
          //调用子类函数 
          fn.apply(this,arguments);
          //复原_super,如果tmp为空就不需要复原了
          tmp && (this._super = tmp);
        }
       })(name,prop[name]) : prop[name];
    }
    //当new一个对象时,实际上是调用该类原型上的init方法,注意通过new调用时传递的参数必须和init函数的参数一一对应
    function Class(){
      if(!initializing && this.init){
        this.init.apply(this,arguments);  
      }
    }    
    //给子类设置原型
    Class.prototype = prototype;
    //给子类设置构造函数
    Class.prototype.constructor = Class;
    //设置子类的extend方法,使得子类也可以通过extend方法被继承
    Class.extend = arguments.callee;
    return Class;
  }
})();

透過使用simple-inheritance函式庫,我們就可以透過很簡單的方式實作繼承了,是不是發現特別像強型別語言的繼承。

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源碼分析,希望本文分享能夠幫助到大家。

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn