摘要:而且在超類型的原型中定義的方法,對子類型而言也是不可見的,結果所有類型都只能使用構造函數模式。在主要考慮對象而不是自定義類型和構造函數的情況下,這個模式也不錯。
寫在前面
注:這個系列是本人對js知識的一些梳理,其中不少內容來自書籍:Javascript高級程序設計第三版和JavaScript權威指南第六版,感謝它們的作者和譯者。有發現什么問題的,歡迎留言指出。
1.原型鏈將原型鏈作為實現繼承的方法,基本思想就是利用原型讓一個引用類型繼承另一個引用類型的屬性和方法:
function SuperType() { this.property = true; } SuperType.prototype.getSuperValue = function () { return this.property; } function SubType() { this.subproperty = false; } //繼承了SuperType SubType.prototype = new SuperType(); SubType.prototype.getSubValue = function () { return this.subproperty; } var instance = new SubType(); console.log(instance.getSuperValue());//true
原型鏈實現繼承的問題:①問題來自包含引用類型值的原型,因為原來的實例屬性變成現在的原型屬性,會被共享,②在創建子類時,不能向超類型的構造函數中傳遞參數。
function SuperType() { this.colors = ["red","blue","green"]; } function SubType() {} //繼承了SuperType SubType.prototype = new SuperType(); var instance1 = new SubType(); instance1.colors.push("black"); console.log(instance1.colors);//["red", "blue", "green", "black"] var instance2 = new SubType(); console.log(instance2.colors);//["red", "blue", "green", "black"]2.借用構造函數
即是在子類型構造函數的內部調用超類型構造函數(還可以傳遞參數):
function SuperType(name) { this.name = name; this.colors = ["red","blue","green"]; } function SubType() { SuperType.call(this,"jaychou"); } var instance1 = new SubType(); instance1.colors.push("black"); console.log(instance1.colors);//["red", "blue", "green", "black"] console.log(instance1.name);//jaychou var instance2 = new SubType(); console.log(instance2.colors);//["red", "blue", "green"] console.log(instance2.name);//jaychou
借用構造函數的問題:方法都在構造函數中定義,沒有進行函數復用。而且在超類型的原型中定義的方法,對子類型而言也是不可見的,結果所有類型都只能使用構造函數模式。考慮到這些問題,借用構造函數的技術也是很少多帶帶使用的。
3.組合繼承即使用原型鏈實現對原型屬性和方法的繼承,而通過借用構造函數來實現對實例屬性的繼承。既通過在原型上定義方法實現了函數復用,又能夠保證每個實例都有它自己的屬性:
function SuperType(name) { this.name = name; this.colors = ["red","blue","green"]; } SuperType.prototype.sayName = function () { console.log(this.name); } function SubType(name,age) { //繼承實例屬性 SuperType.call(this,name); this.age = age; } //繼承原型中的方法 SubType.prototype = new SuperType(); SubType.prototype.constructor = SubType; SubType.prototype.sayAge = function () { console.log(this.age); } var instance1 = new SubType("jaychou",34); instance1.colors.push("black"); console.log(instance1.colors);//["red", "blue", "green", "black"] instance1.sayName();//jaychou instance1.sayAge();//34 var instance2 = new SubType("xiaoming",15); console.log(instance2.colors);//["red", "blue", "green"] instance2.sayName();//xiaoming instance2.sayAge();//154.原型式繼承
沒有使用嚴格意義上的構造函數,借助原型可以基于已有的對象創建新對象的特點。同時還不必因此創建自定義類型:
function inherit(p) { if(p==null) throw TypeError(); if(Object.create) return Object.create(p); var t = typeof p; if(t !== "object" && t !== "funtion") throw TypeError(); function f() {}; f.prototype = p; return new f(); }
好處:簡單直接方便,只是簡單地想一個對象與另一個對象保持類似的情況下,原型式繼承是很不錯的做法。注意點:包含引用類型值的屬性始終都會共享相應的值,就像使用原型模式一樣,如:
var person = { name:"jaychou", friends:[1,2] } var anotherP = inherit(person); anotherP.friends.push(3); console.log(person.friends);//[1, 2, 3] anotherP.name = "xiaoming"; console.log(anotherP.name);//xiaoming console.log(person.name);//jaychou
所以從本質上講,原型式繼承就是對傳過來的對象執行了一次淺復制。
5.寄生式繼承即創建一個僅用于封裝繼承過程的函數,該函數在內部以某種方式來增強對象,最后返回對象(接上):
function createAnother(original) { var clone = inherit(original); clone.sayHi = function () { console.log("hi"); } return clone; } var person = { name:"jaychou", friends:[1,2] }; var anotherPerson = createAnother(person); anotherPerson.sayHi();
在例子里給傳進來的對象增加了方法sayHi,增強了對象。在主要考慮對象而不是自定義類型和構造函數的情況下,這個模式也不錯。缺點:不能做到函數復用而降低效率了。
6.寄生組合式繼承組合繼承的問題就是調用了兩次超類型構造函數:一次是在創建子類型原型的時候,另一次是在子類型構造函數內部。解決方案就是:不必為了指定子類型的原型而調用超類型的構造函數,我們所需要的無非就是超類型原型的一個副本而已。本質上,就是使用寄生式繼承來繼承超類型的原型,然后再將結果指定給子類型的原型:
//寄生組合式繼承 function inheritPrototype(subType, superType) { var prototype = inherit(superType.prototype); prototype.constructor = subType; subType.prototype = prototype; } function SuperType(name) { this.name = name; this.colors = ["red","blue","green"]; } SuperType.prototype.sayName = function () { console.log(this.name); } function SubType(name,age) { //借用超類型的構造函數繼承了超類型的實例屬性 SuperType.call(this,name); this.age = age; } //寄生式繼承了超類型的原型方法 inheritPrototype(SubType,SuperType); SubType.prototype.sayAge = function () { console.log(this.age); } var instance1 = new SubType("jaychou",34); instance1.sayName();//jaychou instance1.sayAge();//34 console.log(SuperType.prototype.isPrototypeOf(instance1));//true console.log(instance1 instanceof SuperType);//true
好處:①只調用了一處SuperType構造函數,②避免了在SubType.prototype上面創建不必要的、多余的屬性。綜上,寄生組合式繼承是繼承的最理想方式。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/108369.html
摘要:起因構造函數對象字面量都可以用來創建單個對象,但有明顯缺點使用同一個接口創建很多對象,會產生大量的重復代碼。組合使用構造函數模式和原型模式創建自定義類型的最常見方式,就是組合使用構造函數模式與原型模式。 寫在前面 注:這個系列是本人對js知識的一些梳理,其中不少內容來自書籍:Javascript高級程序設計第三版和JavaScript權威指南第六版,感謝它們的作者和譯者。有發現什么問題...
摘要:跨域請求詳解從繁至簡前端掘金什么是為什么要用是的一種使用模式,可用于解決主流瀏覽器的跨域數據訪問的問題。異步編程入門道典型的面試題前端掘金在界中,開發人員的需求量一直居高不下。 jsonp 跨域請求詳解——從繁至簡 - 前端 - 掘金什么是jsonp?為什么要用jsonp?JSONP(JSON with Padding)是JSON的一種使用模式,可用于解決主流瀏覽器的跨域數據訪問的問題...
摘要:引言上篇文章介紹原型,這篇文章接著講繼承,嘔心瀝血之作,大哥們點個贊呀明確一點并不是真正的面向對象語言,沒有真正的類,所以我們也沒有類繼承實現繼承有且僅有兩種方式,和原型鏈在介紹繼承前我們先介紹下其他概念函數的三種角色一個函數,有三種角色。 showImg(https://segmentfault.com/img/bVbo4hv?w=1800&h=1000); 引言 上篇文章介紹原型,...
摘要:同理,原型鏈也是實現繼承的主要方式的只是語法糖。原型對象也可能擁有原型,并從中繼承方法和屬性,一層一層以此類推。利用構造函數小明張三張三小明缺點每次實例化都需要復制一遍函數到實例里面。寄生構造函數模式只有被類出來的才能用。 showImg(https://segmentfault.com/img/bVbo4hv?w=1800&h=1000); 引言 最近又攀登了一下JS三座大山中的第二...
摘要:作用域分類作用域共有兩種主要的工作模型。換句話說,作用域鏈是基于調用棧的,而不是代碼中的作用域嵌套。詞法作用域詞法作用域中,又可分為全局作用域,函數作用域和塊級作用域。 一篇鞏固基礎的文章,也可能是一系列的文章,梳理知識的遺漏點,同時也探究很多理所當然的事情背后的原理。 為什么探究基礎?因為你不去面試你就不知道基礎有多重要,或者是說當你的工作經歷沒有亮點的時候,基礎就是檢驗你好壞的一項...
閱讀 3264·2023-04-25 22:47
閱讀 3776·2021-10-11 10:59
閱讀 2310·2021-09-07 10:12
閱讀 4259·2021-08-11 11:15
閱讀 3439·2019-08-30 13:15
閱讀 1757·2019-08-30 13:00
閱讀 975·2019-08-29 14:02
閱讀 1690·2019-08-26 13:57