成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

再論JavaScript原型繼承和對象繼承

Amos / 1969人閱讀

摘要:由于原型即本身也是對象,所以原型繼承可認為是一種特殊的對象式繼承。原型繼承里的原型即是函數(shù)的特有屬性,原型繼承事先得有函數(shù)。揭秘魔術箱不論原型繼承還是對象式繼承,其核心技術是實現(xiàn)了對象實例的鏈。

JavaScript的原型繼承是老生常談。由于原型即prototype本身也是對象,所以“原型”繼承可認為是一種特殊的“對象式”繼承?!睂ο笫健袄^承是筆者基于自己的理解,所提出的一個名詞。本文就著重闡述這兩種繼承方式的異同之處。

原型繼承

JavaScript里的原型即prototype是函數(shù)的特有屬性,原型繼承事先得有函數(shù)。

// 定義函數(shù)Foo
function Foo(name) {
    this.name = name;
}
// 定義Foo的原型
Foo.prototype.say = function() {
    console.log(this.name, "say");
}

函數(shù)及其原型定義好了,就可以使用原型繼承了。

var foo = new Foo("foo");
foo.say() //foo say
foo instanceof Foo; //true。 foo 是Foo的一個實例
foo.__proto__ === Foo.prototype // true

上面用的是new操作符,它實際是通過Object.create工作,其過程如下。所以new其實是Object.create的便利操作方式。

var foo = Object.create(Foo.prototype);
foo.name = "foo"
foo.say();
foo instanceof Foo; //true。 foo 是Foo的一個實例
foo.__proto__ === Foo.prototype // true

請打起精神,Object.create(...)接受一個函數(shù)原型即object實例,以此實例為“模型”,創(chuàng)建新的object實例。新object實例的__proto__指向前function的prototype,自此新object實例也就擁有了前function prototype的字段和方法。

既然Foo.prototype本身是object實例,那么我們是否可以給Object.create(...)傳入一個普通的object實例呢?答案是可以的。這就是本文所要表述的“對象式”繼承。

“對象式“繼承

開門見山地用code來說明:

// 創(chuàng)建對象實例Foo
var Foo = {
    name: "foo",
    say: function() {
        console.log(this.name, "say");
    }
}

// 以Foo為“模型”,創(chuàng)建新的對象實例foo
var foo = Object.create(Foo);

foo.__proto__ == Foo;// true

// 所以foo也會擁有Foo的字段和方法,這點與原型繼承類似。
foo.say(); // foo say. 

// 但是foo不是Foo的實例。
foo instanceof Foo; // TypeError: Right-hand side of "instanceof" is not callable

如果Foo是個函數(shù),結果基本是相同的,除了instanceof Foo 會等于false。

function Foo() {
}
Foo.say = function() {
    console.log(Foo.name, "say");
}

// 以Foo為“模型”,創(chuàng)建新的對象實例foo
var foo = Object.create(Foo);

foo.__proto__ == Foo;// true

// 所以foo也會擁有Foo的字段和方法,這點與原型繼承類似。
foo.say(); // Foo say. 

// 但是foo不是Foo的實例。
foo instanceof Foo; // false
“對象式”繼承的一個應用

舉個不太恰當?shù)睦樱绻覀冃枰獙ath.abs做些“修正”,對于在-1和1之間的數(shù)值保持原值。

 var MMath = Object.create(Math);
 MMath.abs = function(val) {
    if (val >= -1 && val <=1) {
        return val;
    }
    return MMath.__proto__.abs(val); 
 }

 MMath.abs(0.5) // 0.5
 MMath.abs(-0.5) // -0.5
 MMath.abs(-2) // 2
揭秘Object.create魔術箱

不論原型繼承還是“對象式”繼承,其核心技術是Object.create(...)實現(xiàn)了對象實例的__proto__鏈。我們做個簡單的實現(xiàn):

Object.myCreate = function(obj) {
  if (obj instanceof Object || obj === null) {
     var newObj = {};
     Object.setPrototypeOf(newObj, obj)
     return newObj;
  } else {
    throw "error happens. Should input a object";
  } 
}
function Fooo() {}
Fooo.say = function() {
     console.log(Fooo.name, "say");
}

var myFooo = Object.myCreate(Fooo); 
myFooo.say(); // Fooo say

自此我們了解了Object.create(...)的黑魔法,也有助于我們理解Object.create({})和Oject.create(null)的區(qū)別,就是前者的__proto__是個object實例,擁有toString等方法。后者的__proto__是null,它不具有任何方法。在特別注重效率的情景,后者具有優(yōu)勢。

總結

本文提出“對象式”繼承的概念,并與原型繼承對比,闡述其區(qū)別和聯(lián)系,希望有助于深化理解。雖然ES6越來越廣泛應用了,了解函數(shù)原型等這些ES3/ES5的概念應該是有助于JS的深入學習。

文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉載請注明本文地址:http://systransis.cn/yun/84741.html

相關文章

  • 獨家解析Javascript原型繼承

    摘要:面向對象實現(xiàn)代碼動物發(fā)聲汪汪喵喵調用代碼動物發(fā)聲喵喵動物發(fā)聲汪汪當要增加一種動物時,只需增加一個繼承,不會影響其他已有的動物邏輯。所以的繼承和的原型繼承,可謂殊途同歸。 傳統(tǒng)面向對象的繼承和多態(tài) 我們知道C++/Java/C#等面向對象語言,都原生地支持類的繼承。繼承的核心作用大抵是創(chuàng)建一個派生類,并使其復用基本類(即父類)的字段和/或方法。并且派生類可以重寫基本類的方法。這樣基本類和...

    verano 評論0 收藏0
  • [零基礎學python]編寫類之四再論繼承

    摘要:從運行結果可以看出,當子類繼承多個父類的時候,對于構造函數(shù),只有第一個能夠被繼承,第二個就等掉了。重點看,類繼承了,同時,在構造函數(shù)中自己做了規(guī)定,也就是的構造函數(shù)是按照的意愿執(zhí)行,不執(zhí)行的內容,但是,還有一個方法,則繼承了這個方法。 在上一講代碼的基礎上,做進一步修改,成為了如下程序,請看官研習這個程序: #!/usr/bin/env python #coding:utf-8 c...

    BoYang 評論0 收藏0
  • [轉] 再論 java 中的 final 變量

    摘要:的內存分配方式修飾變量通常情況下,變量有個地方可以賦值直接賦值,構造函數(shù)中,或是初始化塊中。如就是對于變量,在聲明時,如果你沒有賦值,系統(tǒng)默認這是一個空白域,在構造函數(shù)進行初始化,如果是靜態(tài)的,則可以在初始化塊。 【java中為什么會有final變量】: final這個關鍵字的含義是這是無法改變的或者終態(tài)的; 那么為什么要阻止改變呢? java語言的發(fā)明者可能由于兩個目的而阻止改變: ...

    soasme 評論0 收藏0
  • Java 多態(tài)(8)

    摘要:在面向對象的程序設計語言中,多態(tài)是繼數(shù)據(jù)抽象和繼承之后的第三種基本特征。 在面向對象的程序設計語言中,多態(tài)是繼數(shù)據(jù)抽象和繼承之后的第三種基本特征。 1.再論向上轉型 多態(tài)作用:消除類型之間的耦合關系. 2.轉機 綁定:將一個方法調用同一個方法主體關聯(lián)起來. 前期綁定:在程序執(zhí)行前就進行綁定(面向過程語言默認綁定方式). 后期綁定:也叫動態(tài)綁定或運行時綁定,在運行時根據(jù)對象的類型進行綁...

    v1 評論0 收藏0
  • 徹底搞懂JavaScript中的繼承

    摘要:這正是我們想要的太棒了毫不意外的,這種繼承的方式被稱為構造函數(shù)繼承,在中是一種關鍵的實現(xiàn)的繼承方法,相信你已經(jīng)很好的掌握了。 你應該知道,JavaScript是一門基于原型鏈的語言,而我們今天的主題 -- 繼承就和原型鏈這一概念息息相關。甚至可以說,所謂的原型鏈就是一條繼承鏈。有些困惑了嗎?接著看下去吧。 一、構造函數(shù),原型屬性與實例對象 要搞清楚如何在JavaScript中實現(xiàn)繼承,...

    _ivan 評論0 收藏0

發(fā)表評論

0條評論

閱讀需要支付1元查看
<