溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

JavaScript繼承的特性與實踐應用深入詳解

發(fā)布時間:2020-09-18 23:14:17 來源:腳本之家 閱讀:126 作者:deniro_li 欄目:web開發(fā)

本文詳細講述了JavaScript繼承的特性與實踐應用。分享給大家供大家參考,具體如下:

繼承是代碼重用的模式。JavaScript 可以模擬基于類的模式,還支持其它更具表現(xiàn)力的模式。但保持簡單通常是最好的策略。

JavaScript 是基于原型的語言,也就是說它可以直接繼承其他對象。

1 偽類

JavaScript 的原型不是直接讓對象從其他對象繼承,而是插入一個多余的間接層:通過構(gòu)造函數(shù)來產(chǎn)生對象。

當一個函數(shù)被創(chuàng)建時,F(xiàn)unction 構(gòu)造器產(chǎn)生的函數(shù)對象會運行這樣類似的代碼:

this.prototype = {constructor : this};

新的函數(shù)對象新增了一個 prototype 屬性,它是一個包含了 constructor 屬性且屬性值為該新函數(shù)的對象。

當采用構(gòu)造器調(diào)用模式,即用 new 去調(diào)用一個函數(shù)時,它會這樣執(zhí)行:

Function.method('new', function (){
  var that = Object.create(this.prototype);//創(chuàng)建一個繼承了構(gòu)造器函數(shù)的原型對象的新對象
  var other = this.apply(that, arguments);//調(diào)用構(gòu)造器函數(shù),綁定 this 到新對象
  return (typeof other === 'object' && other) || that;//如果構(gòu)造器函數(shù)的返回值不是對象,就直接返回這個新對象
});

我們可以定義一個構(gòu)造器,然后擴充它的原型:

//定義構(gòu)造器并擴充原型
var Mammal = function (name) {
  this.name = name;
};
Mammal.prototype.get_name = function () {
  return this.name;
};
Mammal.prototype.says = function () {
  return this.saying || '';
};

然后構(gòu)造實例:

var myMammal = new Mammal('Herb the mammal');
console.log(myMammal.get_name());//Herb the mammal

構(gòu)造另一個偽類來繼承 Mammal(定義構(gòu)造器函數(shù)并替換它的 prototype):

var Cat = function (name) {
  this.name = name;
  this.saying = 'meow';
};
Cat.prototype = new Mammal();

擴充原型:

Cat.prototype.purr = function (n) {
  var i, s = '';
  for (i = 0; i < n; i += 1) {
    if (s) {
      s += '-';
    }
    s += 'r';
  }
  return s;
};
Cat.prototype.get_name = function () {
  return this.says() + ' ' + this.name + ' ' + this.says();
};
var myCat = new Cat('Henrietta');
console.log(myCat.says());//meow
console.log(myCat.purr(5));//r-r-r-r-r
console.log(myCat.get_name());//meow Henrietta meow

我們使用 method 方法定義了 inherits 方法,來隱藏上面這些丑陋的細節(jié):

/**
 * 為 Function.prototype 新增 method 方法
 * @param name 方法名稱
 * @param func 函數(shù)
 * @returns {Function}
 */
Function.prototype.method = function (name, func) {
  if (!this.prototype[name])//沒有該方法時,才添加
    this.prototype[name] = func;
  return this;
};
Function.method('inherits', function (Parent) {
  this.prototype = new Parent();
  return this;
});

這兩個方法都返回 this,這樣我們就可以以級聯(lián)的方式編程啦O(∩_∩)O~

var Cat = function (name) {
  this.name = name;
  this.saying = 'meow';
}.inherits(Mammal).method('purr', function (n) {
    var i, s = '';
    for (i = 0; i < n; i += 1) {
      if (s) {
        s += '-';
      }
      s += 'r';
    }
    return s;
  }).method('get_name', function () {
    return this.says() + ' ' + this.name + ' ' + this.says();
  });
var myCat = new Cat('Henrietta');
console.log(myCat.says());//meow
console.log(myCat.purr(5));//r-r-r-r-r
console.log(myCat.get_name());//meow Henrietta meow

雖然我們有了行為很像“類”的構(gòu)造器函數(shù),但沒有私有環(huán)境,所有的屬性都是公開的,而且不能訪問父類的方法。

如果在調(diào)用構(gòu)造函數(shù)時忘記加上 new 前綴,那么 this 就不會被綁定到新對象上,而是被綁定到了全局變量?。?!這樣我們不但沒有擴充新對象,還破壞了全局變量環(huán)境。

這是一個嚴重的語言設計錯誤!為了降低出現(xiàn)這個問題的概率,所有的構(gòu)造器函數(shù)都約定以首字母大寫的形式來命名。這樣當我們看到首字母大寫的形式的函數(shù),就知道它是構(gòu)造器函數(shù)啦O(∩_∩)O~

當然,更好的策略是根本不使用構(gòu)造器函數(shù)。

2 對象說明符

有時候,構(gòu)造器需要接受一大堆參數(shù),這很麻煩。所以在編寫構(gòu)造器時,讓它接受一個簡單的對象說明符會更好:

var myObject = maker({
 first: f,
 middle: m,
 last: l
});

現(xiàn)在這些參數(shù)可以按照任意的順序排列咯,而且構(gòu)造器還能夠聰明地為那些沒有傳入的參數(shù)使用默認值,代碼也變得更易閱讀啦O(∩_∩)O~

3 原型

基于原型的繼承指的是,一個新對象可以繼承一個舊對象的屬性。首先構(gòu)造出一個有用的對象,然后就可以構(gòu)造出更多與那個對象類似的對象。

/**
 * 原型
 */
var myMammal = {
  name: 'Herb the mammal',
  get_name: function () {
    return this.name;
  },
  says: function () {
    return this.saying || '';
  }
};
//創(chuàng)建新實例
var myCat = Object.create(myMammal);
myCat.name = 'Henrietta';
myCat.saying = 'meow';
myCat.purr = function (n) {
  var i, s = '';
  for (i = 0; i < n; i += 1) {
    if (s) {
      s += '-';
    }
    s += 'r';
  }
  return s;
};
myCat.get_name = function () {
  return this.says() + ' ' + this.name + ' ' + this.says();
};
console.log(myCat.says());//meow
console.log(myCat.purr(5));//r-r-r-r-r
console.log(myCat.get_name());//meow Henrietta meow

這里用到了 create 方法來創(chuàng)建新的實例:

Object.create = function (o) {
    var F = function () {
    };
    F.prototype = o;
    return new F();
 }

4 函數(shù)化

目前為止看到的繼承模式的問題是:無法保護隱私,對象的所有屬性都是可見的。有一些無知的程序員會使用偽裝私有的模式,即給一個需要私有的屬性起一個古怪的名字,并希望其他使用代碼的程序員假裝看不到它們!

其實有更好的方法:應用模塊模式。

我們先構(gòu)造一個生成對象的函數(shù),它有這些步驟:
①. 創(chuàng)建新對象。這有四種方式:
【1】構(gòu)造一個對象字面量。
【2】調(diào)用一個構(gòu)造器函數(shù)。
【3】構(gòu)造一個已存在對象的新實例。
【4】調(diào)用任意一個會返回對象的函數(shù)。
②. 定義私有實例變量與方法。
③. 為這個新對象擴充方法,這些方法擁有特權(quán)去訪問這些參數(shù)。
④. 返回這個新對象。

函數(shù)化構(gòu)造器的偽代碼如下:

var constructor = function (spec, my){
  var that, 其他私有變量;
  my = my || {};
 //把共享的變量和函數(shù)添加到 my 中
 that = 一個新對象
 //添加給 that 的特權(quán)方法
 return that;
};

spec 對象包含了需要構(gòu)造一個新實例的所有信息,它可以被用到到私有變量或者其他函數(shù)中。
my 對象為在一個繼承鏈中的構(gòu)造器提供了共享的容器,如果沒有傳入,那么會創(chuàng)建一個 my 對象。

創(chuàng)建特權(quán)方法的方式是:把函數(shù)定義為私有方法,然后再把它們分配給 that:

var methodical = function (){
 ...
};
that.methodical = methodical;

這樣分兩步定義的好處是:私有的 methodical 不受這個實例被改變的影響。

現(xiàn)在,我們把這個模式應用到 mammal 示例中:

var mammal = function (spec) {
  var that = {};
  that.get_name = function () {
    return spec.name;
  };
  that.says = function () {
    return spec.saying || '';
  };
  return that;
};
var myMammal = mammal({name: 'Herb'});
console.log(myMammal.get_name());//Herb
var cat = function (spec) {
  spec.saying = spec.saying || 'meow';
  var that = mammal(spec);
  that.purr = function (n) {
    var i, s = '';
    for (i = 0; i < n; i += 1) {
      if (s) {
        s += '-';
      }
      s += 'r';
    }
    return s;
  };
  that.get_name = function () {
    return that.says() + ' ' + spec.name + ' ' + that.says();
  };
  return that;
};
var myCat = cat({name: 'Henrietta'});
console.log(myCat.says());//meow
console.log(myCat.purr(5));//r-r-r-r-r
console.log(myCat.get_name());//meow Henrietta meow

函數(shù)化模式還能調(diào)用父類的方法。這里我們構(gòu)造一個 superior 方法,它會返回調(diào)用某個方法名的函數(shù):

//返回調(diào)用某個方法名的函數(shù)
Object.method('superior', function (name) {
  var that = this,
    method = that[name];
  return function () {
    return method.apply(that, arguments);
  };
});

現(xiàn)在創(chuàng)建一個 coolcat,它擁有一個可以調(diào)用父類方法的 get_name:

var coolcat = function (spec) {
  var that = cat(spec),
    super_get_name = that.superior('get_name');
  that.get_name = function (n) {
    return 'like ' + super_get_name() + ' baby';
  };
  return that;
};
var myCoolCat = coolcat({name: 'Bix'});
console.log(myCoolCat.get_name());//like meow Bix meow baby

函數(shù)化模式有很大的靈活性,而且可以更好地實現(xiàn)封裝、信息隱藏以及訪問父類方法的能力。

如果對象所有的狀態(tài)都是私有的,那么就稱為防偽對象。這個對象的屬性可以被替換或刪除,但這個對象的狀態(tài)不受影響。如果用函數(shù)化模式來創(chuàng)建對象,并且這個對象的所有方法都不使用 this 或 that,那么這個對象就是持久性的,它不會被入侵。除非存在特權(quán)方法,否則不能訪問這個持久性對象的內(nèi)部狀態(tài)。

5 事件處理函數(shù)

可以構(gòu)造一個能夠給任何對象添加簡單事件處理特性的函數(shù)。這里,我們給這個對象添加一個 on 方法,fire 方法和私有的事件注冊對象:

var eventuality = function (that) {
  var registry = {};
  /**
   * 觸發(fā)事件
   *
   * 使用 'on' 方法注冊的事件處理程序?qū)⒈徽{(diào)用
   * @param 可以是包含事件名稱的字符串,或者是一個擁有 type 屬性(值為事件名稱)的對象。
   */
  that.fire = function (event) {
    var array,
      func,
      handler,
      i,
      type = typeof event === 'string' ? event : event.type;
    //如果這個事件已被注冊,則遍歷并依序執(zhí)行
    if (registry.hasOwnProperty(type)) {
      array = registry[type];
      for (i = 0; i < array.length; i += 1) {
        handler = array[i];//處理程序包含一個方法和一組可選的參數(shù)
        func = handler.method;
        if (typeof func === 'string') {//如果方法是字符串形式的名稱,則尋找它
          func = this[func];
        }
        //調(diào)用它。如果處理程序包含參數(shù),則傳遞過去,否則就傳遞事件對象
        func.apply(this, handler.parameters || [event]);
      }
    }
    return this;
  };
  /**
   * 注冊一個事件
   * @param type
   * @param method
   * @param parameters
   */
  that.on = function (type, method, parameters) {
    var handler = {
      method: method,
      parameters: parameters
    };
    if (registry.hasOwnProperty(type)) {//如果已存在,就新增數(shù)組項
      registry[type].push(handler);
    } else {//新增
      registry[type] = [handler];
    }
    return this;
  };
  return that;
};

可以在任何單獨對象上調(diào)用 eventuality,授予它事件處理方法。也可以在 that 被返回前,在構(gòu)造函數(shù)中調(diào)用它:

eventuality(that);

JavaScript 弱類型的特性在此是一個巨大的優(yōu)勢,因為我們無須處理對象繼承關(guān)系中的類型O(∩_∩)O~

感興趣的朋友還可以使用本站在線HTML/CSS/JavaScript代碼運行工具:http://tools.jb51.net/code/HtmlJsRun測試上述代碼運行結(jié)果。

更多關(guān)于JavaScript相關(guān)內(nèi)容還可查看本站專題:《javascript面向?qū)ο笕腴T教程》、《JavaScript錯誤與調(diào)試技巧總結(jié)》、《JavaScript數(shù)據(jù)結(jié)構(gòu)與算法技巧總結(jié)》、《JavaScript遍歷算法與技巧總結(jié)》及《JavaScript數(shù)學運算用法總結(jié)》

希望本文所述對大家JavaScript程序設計有所幫助。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI