每個 JavaScript 對象內(nèi)部都有一個指向其它對象的“指針”或者 “引用“, 并通過這種方式在對象之間建立了一種聯(lián)系,形成了一種鏈式結(jié)構(gòu),我的理解這就是所謂的原型鏈。
目前創(chuàng)新互聯(lián)已為超過千家的企業(yè)提供了網(wǎng)站建設(shè)、域名、虛擬主機、綿陽服務器托管、企業(yè)網(wǎng)站設(shè)計、浮山網(wǎng)站維護等服務,公司將堅持客戶導向、應用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶和合作伙伴齊心協(xié)力一起成長,共同發(fā)展。
function?F()?{}?//?F?是一個函數(shù),函數(shù)也是對象,而且每個函數(shù)都有一個屬性叫:"prototype"
var?o?=?new?F();?//?F.prototype?就是?o?的原型對象
console.log(o.name);
//?輸出:?undefined
F.prototype.name?=?"foo";
console.log(o.name);
//?輸出:?foo
//?上面這個例子是想說明:通過?new?在對象?o?與對象?F.prototype?之間建立了聯(lián)系,?這個建立聯(lián)系
//?的方式有人叫?"原型繼承"?。?當訪問的對象屬性不存在時,就會沿著原型鏈去查找。
講原型和原型鏈,如果是講定義,那很是晦澀難懂,今天我們就通俗易懂的說說原型與原型鏈。還需要借助阮老師的“Javascript繼承機制的設(shè)計思想”。
1,比如我們還要針對學生統(tǒng)計每個人的總分是多少,我們改造構(gòu)造函數(shù)Person,構(gòu)造函數(shù)上有個 prototype屬性,這個屬性就是這個構(gòu)造函數(shù)的原型(顯式原型),這個原型是函數(shù)特有,prototype對象默認有兩個屬性,constructor屬性和__proto__屬性。
2,constructor,這個屬性包含了一個指針,指回原構(gòu)造函數(shù)。通過控制臺輸出,我們可以看到我們實例化的對象,有個__proto__屬性,這個屬性就是隱式原型,這個__proto__是所有對象都有的屬性。
3,由于JavaScript的一切都是對象(除undefined),又由于所有對象都有__proto__屬性,__proto__又指向構(gòu)造函數(shù)的prototype,當我們訪問一個對象的屬性時。
4,如果這個對象內(nèi)部不存在這個屬性,那么他就會去__proto__里找這個屬性,這個__proto__又會有自己的__proto__,于是就這樣 一直找下去,也就是我們平時所說的原型鏈的概念。原型鏈,說明是鏈式,而不是環(huán),說明有終點,它的終點是null。
Object() Function() Array() 等等都為構(gòu)造函數(shù)。
Js 面向?qū)ο笈c傳統(tǒng)oop 有些不同,語法較高級 語法糖封裝。
This 為指針。指向this作用域的調(diào)用者
1.原型繼承鏈頂端為Object 。Js函數(shù)是對象
2.當讀取對象中的屬性時,先去實例本身中搜索,如搜索不到則去指向的原型中搜索
1.原型的存在意義在于實現(xiàn)繼承共享,是在構(gòu)造函數(shù)中定義的一個成員對象,在下次實例化時不需要在構(gòu)造函數(shù)中定義成員 就可實現(xiàn)實例共享方法屬性。
例子:通常為。 構(gòu)造函數(shù).prototype.xxx=我想實現(xiàn)實例繼承的東西 -》 new 構(gòu)造函數(shù) -》新實例a對象.原型指針指向構(gòu)造函數(shù)的xxx對象(引用類型)
例子:Array 不等于 Array() 原因 Array為一個函數(shù),而Array()為一個構(gòu)造函數(shù)調(diào)用語句,故Array擁有prototype對象用于實例的共享繼承,Array()產(chǎn)生一個實例 故只能擁有prototype對象的私有指針 proto
2.在使用原型繼承時 不能使用字面量 構(gòu)造函數(shù).prototype={} 的方式重寫原型對象 。因為會導致該原型對象的constructor屬性被重寫,在生成的實例中導致constructor指向Object并且會切斷之前原型對象的聯(lián)系,破壞原型鏈。
3.JavaScript 主要通過原型鏈實現(xiàn)繼承。原型鏈的構(gòu)建是通過將一個類型的實例賦值給另一個構(gòu)造函數(shù)的原型實現(xiàn)的
例子:
xxx實例. proto -》function xxx()構(gòu)造函數(shù).prototype對象故xxx. proto === xxx.prototype
xxx.prototype. proto -》Object.prototype因為所有對象都為Object函數(shù)構(gòu)造來的。故xxx.prototype. proto ===Object.prototype。
Object.prototype. proto 為原型鏈頂端 proto 定義了尚未使用所以為null故Object.prototype. proto ===null約定俗成。
instanceof 用來判斷某實例是否為某構(gòu)造函數(shù)的實例
isPrototypeOf 用于判斷某實例是否擁有某構(gòu)造函數(shù)的原型對象指針
1.原型模式有忽略構(gòu)造函數(shù)定義初始值步驟及原型中操作引用類型的缺點。
所以需要組合使用 構(gòu)造函數(shù)模式+原型模式 來創(chuàng)建實例。在構(gòu)造函數(shù)中定義實例的屬性,而需共享的方法就定義在原型對象中。
繼承:在子構(gòu)造函數(shù)中調(diào)用 父.call(this, name); 實現(xiàn)構(gòu)造函數(shù)之間的屬性繼承。使用 子.prototype = new 父(); 子.prototype.constructor = 子;實現(xiàn)方法的繼承。
2.如要在生產(chǎn)環(huán)境下的構(gòu)造函數(shù)新增方法(如Array) 需要使用一個類似于工廠函數(shù)的寄生構(gòu)造函數(shù)模式 在構(gòu)造函數(shù)中返回一個修改后的對象
javascript原型,原型鏈特點:原型鏈實現(xiàn)了繼承。
JS中每個函數(shù)都存在有一個原型對象屬性prototype。并且所有函數(shù)的默認原型都是Object的實例。每個繼承父函數(shù)的子函數(shù)的對象都包含一個內(nèi)部屬性proto。該屬性包含一個指針,指向父函數(shù)的prototype。若父函數(shù)的原型對象的_proto_屬性為再上一層函數(shù)。在此過程中就形成了原型鏈。
原型鏈的作用是用來實現(xiàn)繼承,比如我們新建一個數(shù)組,數(shù)組的方法就是從數(shù)組的原型上繼承而來的。
var arr = [];
arr.map === Array.prototype.map //arr.map
是從arr.__proto__上繼承下來的,arr.__proto__也就是Array.prototype。
擴展資料:
1.JS中每個函數(shù)都存在有一個原型對象屬性prototype。并且所有函數(shù)的默認原型都是Object的實例。
2.每個繼承父函數(shù)的子函數(shù)的對象都包含一個內(nèi)部屬性_proto_。該屬性包含一個指針,指向父函數(shù)的prototype。若父函數(shù)的原型對象的_proto_屬性為再上一層函數(shù)。在此過程中就形成了原型鏈。
3.原型鏈實現(xiàn)了繼承。原型鏈存在兩個問題:a 包含引用類型值的原型屬性會被所有實例共享。b 在創(chuàng)建子類型時,無法向超類型的構(gòu)造函數(shù)中傳遞參數(shù)。
參考資料:百度百科-javascript
原型是JS的一個引用對象,就是說prototype本身就是一個對象,和OBJECT沒什么不同
所有的屬性中都會包含一個屬性__proto__來指向你的原型對象
例如
function?ROOT(){}
ROOT.A?=?1
ROOT.prototype.B?=?2;
//object對象是默認指向OBJECT直到你指定一個proto指針
//這里由于child本身是已經(jīng)實例化的OBJECT,所以會復制一份指定的root的prototype指針引用到child
var?child?=?{__proto__:ROOT.prototype}
//這里是child1原__proto__是指向空的object
//這里對child1的prototype指針進行了替換
var?child1?=?function?(){};
child1.prototype?=?ROOT.prototype
//剩下的結(jié)果就不說了
var?chd?=?new?child1;
console.log(child.B)
console.log(chd.B)
//你會發(fā)現(xiàn)除了child無法實例化,其他的特性都繼承了
console.log(chd.constructor)
console.log(chd.__proto__)
console.log(child.constructor)
console.log(child.__proto__)
所以這里面
__proto__是指向的創(chuàng)建者的原型對象
constructor是指向的創(chuàng)建者
prototype是當前對象的原型對象
實例化的對象會先從實例化對象里查找屬性,如果找不到會在__proto__里面查找指向的prototype的屬性
我認為,不是不能給父類構(gòu)造函數(shù)傳參數(shù),而是傳的參數(shù)最終不能起到作用。舉個例子:
function Parents(ln) { this.lastName=ln; } //定義父類構(gòu)造函數(shù)
function Children(fn,ln) { this.firstName=fn; } //定義子類,lastName 繼承自父類
//原型鏈繼承,給父類構(gòu)造函數(shù)傳入?yún)?shù),試圖用 Children 類構(gòu)造函數(shù)中傳入的 ln 初始化 lastName:
Children.prototype=new Parents(this ln);
//嘗試建立對象實例:
var child=new Children("Bill","Gates");
//輸出結(jié)果。很明顯,lastNmae 并沒有得到想要的值:
alert(child.firstName);//Bill
alert(child.lastName);//Undefine
這說明給父類構(gòu)造函數(shù)傳遞參數(shù)是無效的。原因就在于原型鏈方式中,調(diào)用父類構(gòu)造函數(shù)的代碼并不在子類構(gòu)造函數(shù)中,建立對象實例時給的屬性值(即子類構(gòu)造函數(shù)的參數(shù))并不能影響到子類調(diào)用的父類構(gòu)造函數(shù)。
當然,在繼承時可以這樣寫:
Children.prototype=new Parents("Gates");//調(diào)用父類構(gòu)造函數(shù)時給固定值
但是,這個固定的屬性值必定會影響所有子類的對象實例,相當于子類構(gòu)造函數(shù)“擅作主張”給所有對象實例的屬性提前“賦了值”。這樣寫是不太符合面向?qū)ο缶幊痰囊?guī)則的。
新聞名稱:javascript原型鏈,原型鏈js簡書
URL鏈接:http://vcdvsql.cn/article36/dsigssg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站制作、電子商務、網(wǎng)站設(shè)計公司、Google、小程序開發(fā)、網(wǎng)站導航
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)