首頁 >web前端 >js教程 >javascript 原型鏈維護與繼承詳解_javascript技巧

javascript 原型鏈維護與繼承詳解_javascript技巧

WBOY
WBOY原創
2016-05-16 16:30:271479瀏覽

一.兩個原型

很多人都知道javascript是原型繼承,每個建構函式都有一個prototype成員,透過它就可以把javascript的繼承演義的美輪美奐了.
其實啊,光靠這一個屬性是無法完成javascript的繼承.
我們在程式碼中使用的prototype完成繼承在這裡就不多說了.大家可以查一下資料.
另外一個看不見的prototype成員.
每一個實例都有有一條指向原型的prototype屬性,這個屬性是無法被訪問到的,當然也就無法被修改了,因為這是維護javascript繼承的基礎.

複製程式碼 程式碼如下:

//建構器宣告
        function Guoyansi(){ }
        function GuoyansiEx(){}
        //原型繼承
         GuoyansiEx.prototype=new Guoyansi();
       //建立物件
       var g1=new GuoyansiEx();
       var g2=new GuoyansiEx();

上面的程式碼中的物件可以用下面的圖來說明

二.原型的維護

一個構造器產生的實例,其constructor屬性總是指向該構造器.我們暫且認為該話是對的.

複製程式碼 程式碼如下:

function Guoyansi(){ }
var obj1=new Guoyansi();
console.log(obj1.constructor===Guoyansi);//true

其實構造器本身是沒有constructor這個屬性的,那麼這個屬性是來自哪呢?
答案是:來自原型.
因此得出下面的結論

複製程式碼 程式碼如下:
obj1.constructor===Guoyansi.prototype.consi.

既然我們可以透過constructor來尋找構造器.因此我們就可以進一步完善上面的圖了.

複製程式碼 程式碼如下:

 function GuoyansiEx(){}
             GuoyansiEx.prototype=new Guoyansi();
             console.log(GuoyansiEx.constructor===GuoyansiEx)//false

根據上圖,上面的結果應該是true,但為什麼是false呢?

現在做個分析.
GuoyansiEx的原型被Guoyansi的實例重寫了,那麼GuoyansiEx的原型中的constructor自然也是來自Guoyansi的實例.
而Guoyansi實例中的constructor又是來自Guoyansi.prototype.而Guoyansi.prototype沒有被重寫,
所以Guoyansi.prototype的constructor指向Guoyansi(建構子);

根據上述分析得出下面的結論

複製程式碼 程式碼如下:
GuoyansiEx.constructor===Guoyansi.constructor===Guoyansi; 🎜>
如果在開發過程中對於Constructor的指向要求非常精確的話,可以做如下處理.

複製程式碼 程式碼如下:
/**方法一:**/
 function Guoyansi(){}
             function GuoyansiEx(){}
             GuoyansiEx.prototype=new Guoyansi();
             GuoyansiEx.prototype.constructor=GuoyansiEx;//重置constructor指向.

複製程式碼 程式碼如下:

/**
            方法二
            **/
            function Guoyansi(){}
            function GuoyansiEx(){
                this.constructor=arguments.callee;
            }
            GuoyansiEx.prototype=new Guoyansi();

複製程式碼 程式碼如下:

/**
            方法三
            **/
            function Guoyansi(){}
            function GuoyansiEx(){
                this.constructor=GuoyansiEx;
            }
            GuoyansiEx.prototype=new Guoyansi();

三.看不見的原型有什麼用呢?

看得見的原型鏈我們可以對他操作來完成我們的繼承,那麼這個看不見的原型鏈我們既看不見,又無法操作.要它有何用.
物件導向中繼承有一個特性:相似性.子類別與父類別具有相似性.因此在子類別中你是無法用delete刪除從父類別繼承而來的成員.也就是說子類別必須具有父類別的特性.
為了維護這個特性,javascript在物件的內部產生了一條我們看不見的原型屬性,並且不允許使用者存取.這樣,使用者可以處於任何目的來修改constructor,
而不會破壞子類別擁有父類別的特性.
簡而言之:內部原型是javascript的原型繼承機制所需要的,而外部原型是用戶實現繼承所需要的.

四.火狐引擎SpiderMonkey中的__proto__

還是這段程式碼.

複製程式碼 程式碼如下:

function Guoyansi(){}
            Guoyansi.prototype.age=24;
            function GuoyansiEx(){}
            var obj1=new Guoyansi();
            GuoyansiEx.prototype=obj1;
            GuoyansiEx.prototype.constructor=GuoyansiEx;//重置constructor指向.
            var obj2=new GuoyansiEx();

我現在想要從obj開始向上訪問父類Guoyansi的prototype的屬性的age.
思路是這樣的.
第一步:obj2====>obj2.constructor.prototype
第二部:obj2.constructor.prototype===>GuoyansiEx.prototype;
第三部:GuoyansiEx.prototype===>obj1;
第四部:obj1.constructor====>Guoyansi
第五部:Guoyansi.prototype.age

寫成這樣:console.log(obj2.constructor.prototype.constructor.prototype.age)//24;
最終的結果是24.
最終的結果是24.可以正常執行,但是在好多書上說constructor修改後,級無法在找到父類中的原型了.不知道是怎麼回事.

在火狐中提夠了一種更簡潔的屬性._proto_
SpiderMonkey中預設在任何建立的物件上新增了一個名為_proto_的屬性,該屬性指向建構器所使用的原型.
其實就是我們上面提到的不可見的原型鏈,只不過是在這個地方變相的公開而已.
可以這樣存取到age
console.log(obj2.__proto__.__proto__.age);//24
這樣的確是成功的訪問到了父類的原型屬性,但是這個屬性只適用於火狐,在其他瀏覽器中是會出錯的.
在E5中對Object做出了擴充Object.getPrototypeOf(),可以存取到所有父類別的原型了.

複製程式碼 程式碼如下:

function Guoyansi(){}
            Guoyansi.prototype.age=24;
            function GuoyansiEx(){}
            var obj1=new Guoyansi();
            GuoyansiEx.prototype=obj1;
            GuoyansiEx.prototype.constructor=GuoyansiEx;//重置constructor指向.
            var obj2=new GuoyansiEx();
            var proto=Object.getPrototypeOf(obj2);
            while(proto){
                console.log(proto.constructor);
                proto=Object.getPrototypeOf(proto);
            }
            console.log("object的原型" proto);

結果是:GuoyansiEx
Guoyansi
Object
object的原型null

個人覺得這些應該算是javascript面向對象的精髓之一了.小伙伴們自己參考下,根據需求使用到自己的項目中去吧

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