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

資訊專欄INFORMATION COLUMN

JS面向?qū)ο蟮某绦蛟O(shè)計(jì)之繼承的實(shí)現(xiàn) - 原型鏈的注意點(diǎn)

pf_miles / 1553人閱讀

摘要:第一種方式是使用操作符,只要檢測(cè)的實(shí)例對(duì)象中的原型鏈包含出現(xiàn)過(guò)的構(gòu)造函數(shù),結(jié)果就會(huì)返回。而這也正是組合使用原型模式和構(gòu)造函數(shù)模式的原因。在構(gòu)造函數(shù)模式中定義屬性,在原型模式中定義共享的方法。

前言:最近在細(xì)讀Javascript高級(jí)程序設(shè)計(jì),對(duì)于我而言,中文版,書(shū)中很多地方翻譯的差強(qiáng)人意,所以用自己所理解的,嘗試解讀下。如有紕漏或錯(cuò)誤,會(huì)非常感謝您的指出。文中絕大部分內(nèi)容引用自《JavaScript高級(jí)程序設(shè)計(jì)第三版》。

接上篇

JS面向?qū)ο蟮某绦蛟O(shè)計(jì)之繼承的實(shí)現(xiàn) - 原型鏈

1. 別忘記默認(rèn)的原型

事實(shí)上,前面例子中展示的原型鏈還少一環(huán)。

我們都知道, 所有引用類型默認(rèn)都繼承了Object,而這個(gè)繼承也是通過(guò)原型鏈實(shí)現(xiàn)的。

所有函數(shù)的默認(rèn)原型是Object的實(shí)例。因?yàn)楹瘮?shù)的原型對(duì)象也是對(duì)象嘛! 對(duì)象當(dāng)然是Object的實(shí)例咯!

因此函數(shù)的原型都會(huì)包含一個(gè)內(nèi)部指針(__proto__), 指向Object.prototype。

這也是所有自定義類型都會(huì)繼承toString()、valueOf()等默認(rèn)方法的根本原因。

所以,上篇例子中展示的原型的原型鏈中還應(yīng)該包括另外一個(gè)繼承層次。

以下代碼展示了這個(gè)完整的原型鏈。

//完整原型鏈的偽代碼

function Object() {
}

Object.prototype = {
    constructor: f Object(),
    hasOwnProperty: f hasOwnProperty(),
    isPrototypeOf: f isPrototypeOf(),
    propertyIsEnumerable: f propertyIsEnumerable(),
    toLocaleString: f toLocaleString(),
    toString: f toString(),
    valueOf: f valueOf()
}

//SuperType 父類型
function SuperType(){
    this.property = true;
}

SuperType.prototype.getSuperProperty = function() {
    console.log(this.property);
    return this.property;
}

/*

SuperType.prototype = {
    constructor: f SuperType(),
    getSuperProperty: function() {
    console.log(this.property);
    return this.property;
    }, 
    __proto__ : {
        constructor: f Object(),
        hasOwnProperty: f hasOwnProperty(),
        isPrototypeOf: f isPrototypeOf(),
        propertyIsEnumerable: f propertyIsEnumerable(),
        toLocaleString: f toLocaleString(),
        toString: f toString(),
        valueOf: f valueOf()
    }
}
*/


//SubType 子類型
function SubType() {
    this.subproperty = false;
}


//子類型 繼承 父類型
SubType.prototype = new SuperType();


//實(shí)際上子類型的原型是這樣的。
/*SubType.prototype = {
    property: true,
    __proto__:  {
        constructor : SuperType,
        getSuperProperty:function() {
            console.log(this.property);
            return this.property;
        }
    }
}
*/

SubType.prototype.getSubProperty = function(){
    console.log(this.subproperty);
    return this.subproperty;
}


//那么現(xiàn)在子類型的原型對(duì)象是這樣的
/*SubType.prototype = {
    property: true,
    getSubProperty: function()  {
    console.log(this.subproperty);
    return this.subproperty;
    },
    __proto__:  {
        constructor : SuperType,
        getSuperProperty:function() {
            console.log(this.property);
            return this.property;
        }
    }
}
*/

var subInstanceObject = new SubType();
console.log(subInstanceObject.getSuperProperty()); // true

一句話,SubType(子類型)繼承了SuperType(父類型),

而SuperType(父類型)繼承了Object(祖先)。

當(dāng)調(diào)用subInstanceObject.toString()時(shí),實(shí)際上調(diào)用的是在保存在Object.prototype中的那個(gè)方法。

2. 確定原型和實(shí)例對(duì)象關(guān)系

可以通過(guò)兩種方式來(lái)確定原型和實(shí)例之間的關(guān)系。

第一種方式是使用instanceof操作符,只要檢測(cè)的實(shí)例對(duì)象中的原型鏈包含出現(xiàn)過(guò)的構(gòu)造函數(shù),結(jié)果就會(huì)返回true。
因?yàn)?,這說(shuō)明他們都參與了,實(shí)例對(duì)象的創(chuàng)建。

console.log(subInstanceObject instanceof Object); // true
console.log(subInstanceObject instanceof SuperType); // true
console.log(subInstanceObject instanceof SubType); // true

由于原型鏈的關(guān)系, 我們可以說(shuō)subIntanceObject是Object、SuperType或SubType中任何一個(gè)類型的實(shí)例。

第二種方式是使用isPrototypeOf()方法。同樣,只要是原型鏈中出現(xiàn)過(guò)的原型,都可以說(shuō)該原型鏈所派生的實(shí)例對(duì)象的原型。

console.log(Object.prototype.isPrototypeOf(subInstanceObject)); //true
console.log(SuperType.prototype.isPrototypeOf(subIntanceObject)); // true
console.log(SubType.prototype.isPrototypeOf(subIntanceObject)); //true
3. 謹(jǐn)慎地定義方法

子類型有時(shí)候需要覆蓋父類型的某個(gè)方法,或者需要添加父類型中不存在的某個(gè)方法。

但不管怎么樣,給原型添加方法的代碼一定要放在替換原型的語(yǔ)句之后。

function SuperType() {
    this.property = true;
}

SuperType.prototype.getSuperValue = function() {
    return this.property;
}

function SubType() {
    this.subproperty = false;
}

//繼承了SuperType
SubType.prototype = new SuperType();

//給原型添加方法的代碼一定要放在替換原型的語(yǔ)句之后
//添加新方法
SubType.prototype.getSubValue = function() {
    return this.subproperty;
}

//重寫(xiě) 超類型中 的 方法
SubType.prototype.getSuperValue = function() {
    return false;
}

var instance = new SubType();
console.log(instance.getSuperValue())

以上代碼中,第一個(gè)方法getSubValue()被添加到了SubType中。
第二個(gè)方法getSuperValue()是原型中已經(jīng)存在的一個(gè)方法。
重寫(xiě)這個(gè)方法將會(huì)子類的原型會(huì)查找到屬于自己的getSuperValue()方法。
當(dāng)通過(guò)SuperType的實(shí)例對(duì)象調(diào)用getSuperValue()時(shí), 還會(huì)繼續(xù)調(diào)用原來(lái)的那個(gè)方法。

再次強(qiáng)調(diào),必須在用SuperType的實(shí)例對(duì)象替換原型之后,再定義兩個(gè)方法。

還有一點(diǎn)需要提醒,即在通過(guò)原型鏈實(shí)現(xiàn)繼承時(shí),不能使用對(duì)象字面量創(chuàng)建原型方法。這樣會(huì)重寫(xiě)原型鏈的。

function SuperType(){
    this.property = true;
}

SuperType.prototype.getSuperValue = function(){
    return this.property;
}

function SubType(){
    this.subproperty = false;
}

//繼承SuperType
SubType.prototype = new SuperType();

/* 
現(xiàn)在的原型
SubType.prototype = {

    property: true,
    __proto__: {
        constructor: SuperType,
        getSuperValue: function() {
            return this.property;
        }
    }
}
*/

//使用對(duì)象字面量語(yǔ)法會(huì)改寫(xiě)原型,導(dǎo)致上一行代碼無(wú)效
// SubType.prototype = new Object();
SubType.prototype = {

    getSubValue: function() {
        return this.subproperty;
    },

    someOtherMethod: function () {
        return false;
    }

    /*,
    __proto__ : {
        constructor: fn Object(),
        .......
    }
    */

}

var instance =  new SubType();
console.log(instance.getSuperValue()); // error: instance.getSuperValue is not a function

以上代碼展示了剛剛把SuperType的實(shí)例對(duì)象賦值給原型,緊接著又將原型替換成一個(gè)對(duì)象字面量而導(dǎo)致的問(wèn)題。

因?yàn)镾ubType的原型其實(shí)保存的是一個(gè)Object的實(shí)例,而非SuperType的實(shí)例對(duì)象,因此這條鏈子斷了。

4. 原型鏈的問(wèn)題

原型鏈雖然很強(qiáng)大,可以用它來(lái)實(shí)現(xiàn)繼承,但是總有缺點(diǎn),世界上不存在萬(wàn)全法。

最主要的問(wèn)題來(lái)自包含引用類型值的原型。

包含引用類型值的原型屬性會(huì)被所有實(shí)例對(duì)象共享。

而這也正是組合使用原型模式和構(gòu)造函數(shù)模式的原因。
在構(gòu)造函數(shù)模式中定義屬性,在原型模式中定義共享的方法。

在通過(guò)原型來(lái)實(shí)現(xiàn)原型繼承時(shí),原型實(shí)際上會(huì)變成另一個(gè)類型的實(shí)例對(duì)象。

原先的實(shí)例對(duì)象屬性,也就變成了現(xiàn)在的原型屬性了。

function SuperType() {
    this.colors = ["red", "green", "blue"];
}

function SubType() {
}

// 子類型繼承父類型
SubType.prototype = new SuperType();

/*
SubType.prototype = {
    colors: ["red", "green", "blue"],
    __proto__: {
        constructor: fn SuperType(),
        .....
    }
}
*/

var instance1 = new SubType();

instance1.colors.push("black");

console.log(instance1.colors); // ["red", "green", "blue", "black"]

var instance2 = new SubType();

console.log(instance2.colors); // ["red", "green", "blue", "black"]

原型鏈的第二個(gè)問(wèn)題在于, 沒(méi)有辦法在不影響所有實(shí)例對(duì)象的情況下,給父類型的構(gòu)造函數(shù)傳遞參數(shù)。

由于上述兩個(gè)問(wèn)題的存在,事件中很少會(huì)多帶帶使用原型鏈。

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

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/98507.html

相關(guān)文章

  • JS面向對(duì)象程序設(shè)計(jì)繼承實(shí)現(xiàn) - 原型

    摘要:簡(jiǎn)單回顧一下構(gòu)造函數(shù)原型和實(shí)例對(duì)象之間的關(guān)系每個(gè)構(gòu)造函數(shù)都有一個(gè)原型對(duì)象。找到生成構(gòu)造函數(shù)的原型對(duì)象的構(gòu)造函數(shù),搜索其原型對(duì)象,找到了。 JS面向?qū)ο蟮某绦蛟O(shè)計(jì)之繼承的實(shí)現(xiàn) - 原型鏈 前言:最近在細(xì)讀Javascript高級(jí)程序設(shè)計(jì),對(duì)于我而言,中文版,書(shū)中很多地方翻譯的差強(qiáng)人意,所以用自己所理解的,嘗試解讀下。如有紕漏或錯(cuò)誤,會(huì)非常感謝您的指出。文中絕大部分內(nèi)容引用自《JavaS...

    zhaochunqi 評(píng)論0 收藏0
  • 如何理解JavaScript原型原型鏈?

    摘要:之前有朋友問(wèn)怎么去理解原型和原型鏈的問(wèn)題。理解原型鏈的小技巧將箭頭視作泛化子類到父類關(guān)系那么圖中所有的虛線將構(gòu)成一個(gè)繼承層級(jí),而實(shí)線表示屬性引用。原型鏈?zhǔn)菍?shí)現(xiàn)繼承的重要方式,原型鏈的形成是真正是靠而非。 之前有朋友問(wèn)怎么去理解原型和原型鏈的問(wèn)題。這個(gè)問(wèn)題,在面試中,很多同學(xué)經(jīng)常都會(huì)遇到。這里給大家講講,方便大家記憶。 JavaScript的特點(diǎn)JavaScript是一門(mén)直譯式腳本...

    xuexiangjys 評(píng)論0 收藏0
  • 如何理解JavaScript原型原型鏈?

    摘要:之前有朋友問(wèn)怎么去理解原型和原型鏈的問(wèn)題。理解原型鏈的小技巧將箭頭視作泛化子類到父類關(guān)系那么圖中所有的虛線將構(gòu)成一個(gè)繼承層級(jí),而實(shí)線表示屬性引用。原型鏈?zhǔn)菍?shí)現(xiàn)繼承的重要方式,原型鏈的形成是真正是靠而非。 之前有朋友問(wèn)怎么去理解原型和原型鏈的問(wèn)題。這個(gè)問(wèn)題,在面試中,很多同學(xué)經(jīng)常都會(huì)遇到。這里給大家講講,方便大家記憶。 JavaScript的特點(diǎn)JavaScript是一門(mén)直譯式腳本...

    adie 評(píng)論0 收藏0
  • SegmentFault 技術(shù)周刊 Vol.32 - 七夕將至,你對(duì)象”還好嗎?

    摘要:很多情況下,通常一個(gè)人類,即創(chuàng)建了一個(gè)具體的對(duì)象。對(duì)象就是數(shù)據(jù),對(duì)象本身不包含方法。類是相似對(duì)象的描述,稱為類的定義,是該類對(duì)象的藍(lán)圖或原型。在中,對(duì)象通過(guò)對(duì)類的實(shí)體化形成的對(duì)象。一類的對(duì)象抽取出來(lái)。注意中,對(duì)象一定是通過(guò)類的實(shí)例化來(lái)的。 showImg(https://segmentfault.com/img/bVTJ3H?w=900&h=385); 馬上就要到七夕了,離年底老媽老爸...

    李昌杰 評(píng)論0 收藏0
  • SegmentFault 技術(shù)周刊 Vol.32 - 七夕將至,你對(duì)象”還好嗎?

    摘要:很多情況下,通常一個(gè)人類,即創(chuàng)建了一個(gè)具體的對(duì)象。對(duì)象就是數(shù)據(jù),對(duì)象本身不包含方法。類是相似對(duì)象的描述,稱為類的定義,是該類對(duì)象的藍(lán)圖或原型。在中,對(duì)象通過(guò)對(duì)類的實(shí)體化形成的對(duì)象。一類的對(duì)象抽取出來(lái)。注意中,對(duì)象一定是通過(guò)類的實(shí)例化來(lái)的。 showImg(https://segmentfault.com/img/bVTJ3H?w=900&h=385); 馬上就要到七夕了,離年底老媽老爸...

    Lyux 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

pf_miles

|高級(jí)講師

TA的文章

閱讀更多
最新活動(dòng)
閱讀需要支付1元查看
<