亚洲中字慕日产2020,大陆极品少妇内射AAAAAA,无码av大香线蕉伊人久久,久久精品国产亚洲av麻豆网站

資訊專欄INFORMATION COLUMN

深入javascript——原型鏈和繼承

UCloud / 835人閱讀

摘要:在使用原型鏈實(shí)現(xiàn)繼承時(shí)有一些需要我們注意的地方注意繼承后的變化。在了解原型鏈時(shí),不要忽略掉在末端還有默認(rèn)的對象,這也是我們能在所有對象中使用等對象內(nèi)置方法的原因。

  

在上一篇post中,介紹了原型的概念,了解到在javascript中構(gòu)造函數(shù)、原型對象、實(shí)例三個(gè)好基友之間的關(guān)系:每一個(gè)構(gòu)造函數(shù)都有一個(gè)“守護(hù)神”——原型對象,原型對象心里面也存著一個(gè)構(gòu)造函數(shù)的“位置”,兩情相悅,而實(shí)例呢卻又“暗戀”著原型對象,她也在心里留存了一個(gè)原型對象的位置。

javascript本身不是面向?qū)ο蟮恼Z言,而是基于對象的語言,對于習(xí)慣了其他OO語言的人來說,起初有些不適應(yīng),因?yàn)樵谶@里沒有“類”的概念,或者說“類”和“實(shí)例”不區(qū)分,更不要指望有“父類”、“子類”之分了。那么,javascript中這一堆對象這么聯(lián)系起來呢?
幸運(yùn)的是,javascript在設(shè)計(jì)之初就提供了“繼承”的實(shí)現(xiàn)方式,在認(rèn)識(shí)“繼承”之前,我們現(xiàn)在先來了解下原型鏈的概念。

原型鏈

我們知道原型都有一個(gè)指向構(gòu)造函數(shù)的指針,假如我們讓SubClass原型對象等于另一個(gè)類型的實(shí)例new SuperClass()會(huì)怎么樣?此時(shí),SubClass原型對象包含一個(gè)指向SuperClass原型的指針,SuperClass原型中也包含一個(gè)指向SuperClass構(gòu)造函數(shù)的指針。。。這樣層層遞進(jìn)下去,就形成了一個(gè)原型鏈

具體代碼如下:

    function SuperClass(){
        this.name = "women"
    }
    SuperClass.prototype.sayWhat = function(){
        return this.name + ":i`m a girl!";
    }
    function SubClass(){
        this.subname = "your sister";
    }
    SubClass.prototype = new SuperClass();
    SubClass.prototype.subSayWhat = function(){
        return this.subname + ":i`m a beautiful girl";
    }
    var sub = new SubClass();
    console.log(sub.sayWhat());//women:i`m a girl!
使用原型鏈實(shí)現(xiàn)繼承

通過上面的代碼中可以看出SubClass繼承了SuperClass的屬性和方法,這個(gè)繼承的實(shí)現(xiàn)是通過將SuperClass的實(shí)例賦值給SubClass的原型對象,這樣SubClass的原型對象就被SuperClass的一個(gè)實(shí)例覆蓋掉了,擁有了它的全部屬性和方法,同時(shí)還擁有一個(gè)指向SuperClass原型對象的指針。
在使用原型鏈實(shí)現(xiàn)繼承時(shí)有一些需要我們注意的地方:

注意繼承后constructor的變化。此處sub的constructor指向的是SuperClass,因?yàn)镾ubClass的原型指向了SuperClass的原型。在了解原型鏈時(shí),不要忽略掉在末端還有默認(rèn)的Object對象,這也是我們能在所有對象中使用toString等對象內(nèi)置方法的原因。

通過原型鏈實(shí)現(xiàn)繼承時(shí),不能使用字面量定義原型方法,因?yàn)檫@樣會(huì)重寫原型對象(在上一篇post中也介紹過):

    function SuperClass(){
        this.name = "women"
    }
    SuperClass.prototype.sayWhat = function(){
        return this.name + ":i`m a girl!";
    }
    function SubClass(){
        this.subname = "your sister";
    }
    SubClass.prototype = new SuperClass();
    SubClass.prototype = {//此處原型對象被覆蓋,因?yàn)闊o法繼承SuperClass屬性和方法
        subSayWhat:function(){
            return this.subname + ":i`m a beautiful girl";
        }
    }
    var sub = new SubClass();
    console.log(sub.sayWhat());//TypeError: undefined is not a function

實(shí)例共享的問題。在前面講解原型和構(gòu)造函數(shù)時(shí),我們曾經(jīng)介紹過包含引用類型屬性的原型會(huì)被所有的實(shí)例共享,同樣,我們繼承而來的原型中也會(huì)共享“父類”原型中引用類型的屬性,當(dāng)我們通過原型繼承修改了“父類”的引用類型屬性后,其他所有繼承自該原型的實(shí)例都會(huì)受到影響,這不僅浪費(fèi)了資源,也是我們不愿看到的現(xiàn)象:

    function SuperClass(){
        this.name = "women";
        this.bra = ["a","b"];
    }
    function SubClass(){
        this.subname = "your sister";
    }
    SubClass.prototype = new SuperClass();
    var sub1 = new SubClass();
    sub1.name = "man";
    sub1.bra.push("c");
    console.log(sub1.name);//man
    console.log(sub1.bra);//["a","b","c"]
    var sub2 = new SubClass();
    console.log(sub1.name);//woman
    console.log(sub2.bra);//["a","b","c"]

注意:此處在數(shù)組中添加一個(gè)元素,所有繼承自SuperClass的實(shí)例都會(huì)受到影響,但是如果修改name屬性則不會(huì)影響到其他的實(shí)例,這是因?yàn)閿?shù)組為引用類型,而name為基本類型。
如何解決實(shí)例共享的問題呢?我們接著往下看...

經(jīng)典繼承(constructor stealing)

正如我們介紹過很少多帶帶使用原型定義對象一樣,在實(shí)際開發(fā)中我們也很少多帶帶使用原型鏈,為了解決引用類型的共享問題,javascript開發(fā)者們引入了經(jīng)典繼承的模式(也有人稱為借用構(gòu)造函數(shù)繼承),它的實(shí)現(xiàn)很簡單就是在子類型構(gòu)造函數(shù)中調(diào)用超類型的構(gòu)造函數(shù)。我們需要借助javascript提供的call()或者apply()函數(shù),我們看下示例:

function SuperClass() {
    this.name = "women";
    this.bra = ["a", "b"];
}
function SubClass() {
    this.subname = "your sister";
    //將SuperClass的作用域賦予當(dāng)前構(gòu)造函數(shù),實(shí)現(xiàn)繼承
    SuperClass.call(this);
}

var sub1 = new SubClass();
sub1.bra.push("c");
console.log(sub1.bra);//["a","b","c"]
var sub2 = new SubClass();
console.log(sub2.bra);//["a","b"]

SuperClass.call(this);這一句話的意思是在SubClass的實(shí)例(上下文)環(huán)境中調(diào)用了SuperClass構(gòu)造函數(shù)的初始化工作,這樣每一個(gè)實(shí)例就會(huì)有自己的一份bra屬性的副本了,互不產(chǎn)生影響了。
但是,這樣的實(shí)現(xiàn)方式仍不是完美的,既然引入了構(gòu)造函數(shù),那么同樣我們也面臨著上篇中講到的構(gòu)造函數(shù)存在的問題:如果在構(gòu)造函數(shù)中有方法的定義,那么對于沒一個(gè)實(shí)例都存在一份多帶帶的Function引用,我們的目的其實(shí)是想共用這個(gè)方法,而且我們在超類型原型中定義的方法,在子類型實(shí)例中是無法調(diào)用到的:

    function SuperClass() {
        this.name = "women";
        this.bra = ["a", "b"];
    }
    SuperClass.prototype.sayWhat = function(){
        console.log("hello");
    }
    function SubClass() {
        this.subname = "your sister";
        SuperClass.call(this);
    }   
    var sub1 = new SubClass();
    console.log(sub1.sayWhat());//TypeError: undefined is not a function

如果你看過上篇文章關(guān)于原型對象和構(gòu)造函數(shù)的,想必你已經(jīng)知道解決這個(gè)問題的答案了,那就是沿用上篇的套路,使用“組合拳”!

組合式繼承

組合式繼承就是結(jié)合原型鏈和構(gòu)造函數(shù)的優(yōu)勢,發(fā)出各自特長,組合起來實(shí)現(xiàn)繼承的一種方式,簡單來說就是使用原型鏈繼承屬性和方法,使用借用構(gòu)造函數(shù)來實(shí)現(xiàn)實(shí)例屬性的繼承,這樣既解決了實(shí)例屬性共享的問題,也讓超類型的屬性和方法得到繼承:

    function SuperClass() {
        this.name = "women";
        this.bra = ["a", "b"];
    }
    SuperClass.prototype.sayWhat = function(){
        console.log("hello");
    }
    function SubClass() {
        this.subname = "your sister";
        SuperClass.call(this);             //第二次調(diào)用SuperClass
    }
    SubClass.prototype = new SuperClass(); //第一次調(diào)用SuperClass
    var sub1 = new SubClass();
    console.log(sub1.sayWhat());//hello

組合繼承的方式也是實(shí)際開發(fā)中我們最常用的實(shí)現(xiàn)繼承的方式,到此已經(jīng)可以滿足你實(shí)際開發(fā)的需求了,但是人對完美的追求是無止境的,那么,必然會(huì)有人對這個(gè)模式“吹毛求疵”了:你這個(gè)模式調(diào)用了兩次超類型的構(gòu)造函數(shù)耶!兩次耶。。。你造嗎,這放大一百倍是多大的性能損失嗎?
最有力的反駁莫過于拿出解決方案,好在開發(fā)者找到了解決這個(gè)問題的最優(yōu)方案:

寄生組合式繼承

在介紹這個(gè)繼承方式前,我們先了解下寄生構(gòu)造函數(shù)的概念,寄生構(gòu)造函數(shù)類似于前面提到的工廠模式,它的思想是定義一個(gè)公共函數(shù),這個(gè)函數(shù)專門用來處理對象的創(chuàng)建,創(chuàng)建完成后返回這個(gè)對象,這個(gè)函數(shù)很像構(gòu)造函數(shù),但構(gòu)造函數(shù)是沒有返回值的:

function Gf(name,bra){
    var obj = new Object();
    obj.name = name;
    obj.bra = bra;
    obj.sayWhat = function(){
        console.log(this.name);
    }
    return obj;
}

var gf1 = new Gf("bingbing","c++");
console.log(gf1.sayWhat());//bingbing

寄生式繼承的實(shí)現(xiàn)和寄生式構(gòu)造函數(shù)類似,創(chuàng)建一個(gè)不依賴于具體類型的“工廠”函數(shù),專門來處理對象的繼承過程,然后返回繼承后的對象實(shí)例,幸運(yùn)的是這個(gè)不需要我們自己實(shí)現(xiàn),道哥(道格拉斯)早已為我們提供了一種實(shí)現(xiàn)方式:

function object(obj) {
    function F() {}
    F.prototype = obj;
    return new F();
}
var superClass = {
    name:"bingbing",
    bra:"c++"
}
var subClass = object(superClass);
console.log(subClass.name);//bingbing

在公共函數(shù)中提供了一個(gè)簡單的構(gòu)造函數(shù),然后將傳進(jìn)來對象的實(shí)例賦予構(gòu)造函數(shù)的原型對象,最后返回該構(gòu)造函數(shù)的實(shí)例,很簡單,但療效很好,不是嗎?這個(gè)方式被后人稱為“原型式繼承”,而寄生式繼承正是在原型式基礎(chǔ)上,通過增強(qiáng)對象的自定義屬性實(shí)現(xiàn)的:

function buildObj(obj){
    var o = object(obj);
    o.sayWhat = function(){
        console.log("hello");
    }
    return o;
}
var superClass = {
    name:"bingbing",
    bra:"c++"
}
var gf = buildObj(superClass);
gf.sayWhat();//hello

寄生式繼承方式同樣面臨著原型中函數(shù)復(fù)用的問題,于是,人們又開始拼起了積木,誕生了——寄生組合式繼承,目的是解決在指定子類型原型時(shí)調(diào)用父類型構(gòu)造函數(shù)的問題,同時(shí),達(dá)到函數(shù)的最大化復(fù)用。基于以上基礎(chǔ)實(shí)現(xiàn)方式如下:

//參數(shù)為兩個(gè)構(gòu)造函數(shù)
function inheritObj(sub,sup){
    //實(shí)現(xiàn)實(shí)例繼承,獲取超類型的一個(gè)副本
    var proto = object(sup.prototype);
    //重新指定proto實(shí)例的constructor屬性
    proto.constructor = sub;
    //將創(chuàng)建的對象賦值給子類型的原型
    sub.prototype = proto;
}
function SuperClass() {
    this.name = "women";
    this.bra = ["a", "b"];
}
SuperClass.prototype.sayWhat = function() {
    console.log("hello");
}

function SubClass() {
    this.subname = "your sister";
    SuperClass.call(this);
}
inheritObj(SubClass,SuperClass);
var sub1 = new SubClass();
console.log(sub1.sayWhat()); //hello

這個(gè)實(shí)現(xiàn)方式避免了超類型的兩次調(diào)用,而且也省掉了SubClass.prototype上不必要的屬性,同時(shí)還保持了原型鏈,到此真正的結(jié)束了繼承之旅,這個(gè)實(shí)現(xiàn)方式也成為了最理想的繼承實(shí)現(xiàn)方式!人們對于javascript的繼承的爭議還在繼續(xù),有人提倡OO,有人反對在javascript做多余的努力去實(shí)現(xiàn)OO的特性,管他呢,至少又深入了解了些!

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

轉(zhuǎn)載請注明本文地址:http://www.ezyhdfw.cn/yun/87590.html

相關(guān)文章

  • 剖析JS的原型鏈和繼承

    摘要:接下來我們來聊一下的原型鏈繼承和類。組合繼承為了復(fù)用方法,我們使用組合繼承的方式,即利用構(gòu)造函數(shù)繼承屬性,利用原型鏈繼承方法,融合它們的優(yōu)點(diǎn),避免缺陷,成為中最常用的繼承。 JavaScript是一門面向?qū)ο蟮脑O(shè)計(jì)語言,在JS里除了null和undefined,其余一切皆為對象。其中Array/Function/Date/RegExp是Object對象的特殊實(shí)例實(shí)現(xiàn),Boolean/N...

    darkerXi 評論0 收藏0
  • 【進(jìn)階2-1期】深入淺出圖解作用域鏈和閉包

    摘要:本期推薦文章從作用域鏈談閉包,由于微信不能訪問外鏈,點(diǎn)擊閱讀原文就可以啦。推薦理由這是一篇譯文,深入淺出圖解作用域鏈,一步步深入介紹閉包。作用域鏈的頂端是全局對象,在全局環(huán)境中定義的變量就會(huì)綁定到全局對象中。 (關(guān)注福利,關(guān)注本公眾號回復(fù)[資料]領(lǐng)取優(yōu)質(zhì)前端視頻,包括Vue、React、Node源碼和實(shí)戰(zhàn)、面試指導(dǎo)) 本周開始前端進(jìn)階的第二期,本周的主題是作用域閉包,今天是第6天。 本...

    levius 評論0 收藏0
  • 深入淺出面向?qū)ο蠛?em>原型【概念篇3】—— 原型鏈和繼承

    摘要:由一個(gè)問題引發(fā)的思考這個(gè)方法是從哪兒蹦出來的首先我們要清楚數(shù)組也是對象,而且是對象的實(shí)例也就是說,下面兩種形式是完全等價(jià)的只不過是一種字面量的寫法,在深入淺出面向?qū)ο蠛驮透拍钇恼吕?,我們提到過類會(huì)有一個(gè)屬性,而這個(gè)類的實(shí)例可以通過屬性訪 1.由一個(gè)問題引發(fā)的思考 let arr1 = [1, 2, 3] let arr2 = [4, 5, 6] arr1.c...

    levinit 評論0 收藏0
  • javascript系列--javascript深入淺出圖解作用域鏈和閉包

    摘要:變量對象也是有父作用域的。作用域鏈的頂端是全局對象。當(dāng)函數(shù)被調(diào)用的時(shí)候,作用域鏈就會(huì)包含多個(gè)作用域?qū)ο?。?dāng)函數(shù)要訪問時(shí),沒有找到,于是沿著作用域鏈向上查找,在的作用域找到了對應(yīng)的標(biāo)示符,就會(huì)修改的值。 一、概要 對于閉包的定義(紅寶書P178):閉包就是指有權(quán)訪問另外一個(gè)函數(shù)的作用域中的變量的函數(shù)。 關(guān)鍵點(diǎn): 1、閉包是一個(gè)函數(shù) 2、能夠訪問另外一個(gè)函數(shù)作用域中的變量 二、閉包特性 對...

    Jensen 評論0 收藏0
  • 理解原型鏈和原型繼承

    摘要:原型鏈和構(gòu)造函數(shù)是一種面向?qū)ο蟮恼Z言,并且可以進(jìn)行原型繼承。來了極大的支持了工程化,它的標(biāo)準(zhǔn)讓瀏覽器內(nèi)部實(shí)現(xiàn)類和類的繼承構(gòu)造函數(shù)構(gòu)造函數(shù)調(diào)用父類構(gòu)造函數(shù)現(xiàn)在瀏覽器對其支持程度還不高。 原型鏈 原型鏈比作用域鏈要好理解的多。 JavaScript中的每個(gè)對象,都有一個(gè)內(nèi)置的_proto_屬性。這個(gè)屬性是編程不可見的(雖然ES6標(biāo)準(zhǔn)中開放了這個(gè)屬性,然而瀏覽器對這個(gè)屬性的可見性的支持不同)...

    XboxYan 評論0 收藏0

發(fā)表評論

0條評論

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