摘要:思路是,使用原型鏈對原型屬性和方法進行繼承,借用構造函數實現對實例屬性的繼承。注意使用寄生式繼承來為對象添加函數,會由于不能做到函數復用而降低效率,這一點與構造函數模式類似。無論什么情況下都會調用兩次超類型的構造函數。
說明:
此摘要筆記系列是我最近看《JavaScript高級程序設計(第3版)》隨手所記。
里面分條列舉了一些我認為重要的、需要記下的、對我有幫助的點,是按照我看的順序來的。
摘要筆記本身沒有系統性,沒有全面性可言,寫在這里供有一定基礎的前端開發者參考交流。
里面的知識點、例子大部分來源于本書,或者延伸出來的,都經過我的測試是對的,但是沒辦法保證100%正確,如果有人看到錯誤的地方,希望指出來,謝謝。
function Grandpa () { this.name = "Grandpa" } Grandpa.prototype.sayName = function () { console.log(this.name) } function Parent () { this.name = "Parent" } Parent.prototype = new Grandpa() Parent.prototype.sayHello = function () { console.log("hello") } var ming = new Parent() var li = new Parent() ming.sayName() // "Parent" ming.sayHello() // "hello" ming.sayHello === li.sayHello // true ming.sayName === li.sayName // true ming instanceof Parent // true ming instanceof Grandpa // true ming.constructor === Parent // false 因為在上面重新了 Parent 的原型,而且沒有重新指定 constructor ming.constructor === Grandpa // true 因為上面沒有重寫 Grandpa 的原型 ming.__proto__ === Parent.prototype // true ming.__proto__ === Grandpa.prototype // false Parent.prototype.__proto__ === Grandpa.prototype // true2. 確定原型和實例的關系
兩種方法: instanceof 操作符和 isPrototypeOf(),
如上例:
Parent.prototype.isPrototypeOf(ming) // true Grandpa.prototype.isPrototypeOf(ming) // true Object.prototype.isPrototypeOf(ming) // true3. 給原型添加方法的代碼一定放在替換原型的語句之后,否則無效 4. 原型鏈的問題
function Grandpa () { this.friends = ["ming", "li"] } function Parent () { } Parent.prototype = new Grandpa() var ming = new Parent() var li = new Parent() ming.friends.push("wang") console.log(ming.friends) // ["ming", "li", "wang"] console.log(li.friends) // ["ming", "li", "wang"] ming.friends === li.friends // true
可以解決引用類型的繼承在多個實例之間共享的問題
如:
function Grandpa () { this.friends = ["ming", "li"] } function Aunt () { Grandpa.call(this) } var ming = new Aunt() var li = new Aunt() ming.friends // ["ming", "li"] ming.friends.push("wang") li.friends // ["ming", "li"] ming.friends // ["ming", "li", "wang"]7. 借用構造函數可以解決不能向超類型的構造函數中傳遞參數的問題。
如:
function Grandpa (name) { this.name = name } function Parent (name, age) { Grandpa.call(this, name) this.age = age } var ming = new Parent("ming", 15) ming.age // 15 ming.name // "ming"
注意:這種情況,為了避免超類型的構造函數(Grandpa)重寫子類型(Parnt)的屬性,應在調用超類型構造函數后,再添加應該在子類型中定義的屬性
8. 借用構造函數的問題。問題1. 函數沒有復用,占用內存多。
問題2. 超類型原型中定義的方法,對子類型而言是不可見的,結果只能使用構造函數模式。
有時候也叫偽經典繼承。思路是,使用原型鏈對原型屬性和方法進行繼承,借用構造函數實現對實例屬性的繼承。
如:
function Grandpa (name) { this.name = name this.friends = ["li"] } Grandpa.prototype.sayName = function () { console.log(this.name) } function Parent (name, age) { Grandpa.call(this, name) this.age = age } Parent.prototype = new Grandpa() Parent.prototype.sayAge = function () { console.log(this.age) } var ming = new Parent("ming", 12) var wang = new Parent("wang", 15) ming.name // "ming" ming.friends // ["li"] ming.friends.push("zhang") wang.friends // ["li"] ming.sayName() // "ming" ming.sayAge() // 12 ming.sayName === wang.sayName // true ming.sayAge === wang.sayAge // true ming instanceof Parent // true ming instanceof Grandpa // true10. 原型式繼承
function object (o) { function F () {} F.prototype = o return new F() }
var person = { name: "wang", friends: ["li"] } var person1 = object(person) person1.name // "wang" person1.friends // ["li"] person1.name = "zhang" person.name // "wang" var person2 = object(person) person1.friends.push("yang") person2.friends // ["li", "yang"] person2.friends === person.friends // true person1.friends === person.friends // true
這種原型式繼承,必須有一個對象作為另一個對象的基礎。把基礎對象傳入 object() 函數,把得到的對象根據具體需求做修改即可。
但是別忘了所有得到的對象引用類型是共享的。
這個方法接受兩個參數,一個用作新對象的原型對象,一個(可選的)為新對象定義額外屬性的對象。
在傳入一個參數的情況下, Object.create() 和 object() 的行為相同。
Object.create()的第二個參數與 Object.defineProperties() 的第二個參數格式相同:每個屬性都是通過自己的描述符來定義的。
這種方式指定的任何屬性都會覆蓋原型對象上的同名屬性。
如:
var person = { name: "wang", friends: ["li"] } var person1 = Object.create(person, { name: { value: "zhang" }, friends: { value: ["a", "b"] } }) person1.name // "zhang" person1.friends // ["a", "b"] person.friends // ["li"] // 支持 Object.create() 的瀏覽器有 IE 9+,Firefox 4+, Safari 5+, Opera 912+, Chrome.11. 寄生式繼承
寄生式繼承是與原型式繼承緊密相關的一種思路,和工廠模式類似。如:
function createAnother (original) { var clone = object(original) clone.sayHi = function () { console.log("hi") } return clone } var person = { name: "wang", friends: ["li"] } var person1 = createAnother(person) person1.friends // ["li"] person1.sayHi() // "hi"
新的對象person1不僅具有person的所有屬性和方法,還有自己的 sayHi() 方法。
示范的 object() 函數不是必須的,任何能夠返回新對象的函數都適用于此模式。
注意:使用寄生式繼承來為對象添加函數,會由于不能做到函數復用而降低效率,這一點與構造函數模式類似。
無論什么情況下都會調用兩次超類型的構造函數。一次是在創建子類型原型的時候,一次是在子類型構造函數內部。
如:
function Grandpa (name) { this.name = name this.friends = ["ming"] } Grandpa.prototype.sayName = function () { console.log(this.name) } function Parent (name, age) { Grandpa.call(this, name) this.age = age } Parent.prototype = new Grandpa() // 第一次調用超類型的構造函數 Grandpa(), // 此時原型上有一個friends屬性(值是["ming"])和name屬性(值undefined,因為沒有傳參) Parent.prototype.sayHi = function () { console.log("hi") } var wang = new Parent("wang", 15) // 第二次調用超類型的構造函數 Grandpa(),在 Parent() 函數內部調用, // 這一步的操作, Parent 的實例 wang 的屬性會覆蓋 Parent 的原型上的同名屬性 name12. 寄生組合式繼承
基本模式如下:
function inheritPrototype (subType, superType) { var prototype = object(superType.prototype) prototype.constructor = subType subType.prototype = prototype } function Grandpa (name) { this.name = name this.friends = ["ming"] } Grandpa.prototype.sayName = function () { console.log(this.name) } function Parent (name, age) { Grandpa.call(this, name) this.age = age } inheritPrototype(Parent, Grandpa) // 這里沒有調用超類型的構造函數 Grandpa() ,但是這里調用了一個函數內部的構造函數 Parent.prototype.sayHi = function () { console.log("hi") } var li = new Parent("li", 20) // 調用超類型的構造函數 Grandpa(),在 Parent() 函數內部調用
這種模式的好處是只調用一次 Grandpa() 構造函數,因此避免了在 Parent.prototype 上創建不必要的、多余的屬性。
作者講,開發人員普遍認為寄生組合式繼承是引用類型最理想的繼承范式。
繼承部分全部結束
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/91656.html
摘要:說明此摘要筆記系列是我最近看高級程序設計第版隨手所記。摘要筆記本身沒有系統性,沒有全面性可言,寫在這里供有一定基礎的前端開發者參考交流。對每一項運行給定函數,返回該函數會返回的項組成的數組。是的反操作是的反操作第一部分結束。 說明: 此摘要筆記系列是我最近看《JavaScript高級程序設計(第3版)》隨手所記。 里面分條列舉了一些我認為重要的、需要記下的、對我有幫助的點,是按照我看...
摘要:如果重設構造函數的原型對象,那么,會切斷新的原型對象和任何之前已經存在的構造函數實例之間的聯系,它們引用的仍然是最初的原型。說明返回的對象與構造函數或者與構造函數的原型屬性沒有關系。 說明: 此摘要筆記系列是我最近看《JavaScript高級程序設計(第3版)》隨手所記。里面分條列舉了一些我認為重要的、需要記下的、對我有幫助的點,是按照我看的順序來的。摘要筆記本身沒有系統性,沒有全面性...
摘要:關于對象定義了全局對象。支持的瀏覽器有除了接受要序列化的對象外,還可以接受另外兩個參數。如果是數值,則表示每個級別縮進的空格數,最大,超過的值自動轉換成。字符串長度超過,結果中將只出現前個字符。會在結果字符串中插入換行符提高可讀性。 關于JSON 1. JSON 對象 es5 定義了全局對象 JSON。支持的瀏覽器有 IE8+ 、Firefox 3.5+ 、Safari 4+、Chro...
摘要:函數表達式和閉包函數聲明的一個重要特征是函數聲明提升如遞歸遞歸函數是在一個函數通過名字調用自身的情況下構成的。注意中已經是塊級作用域了,所以這些東西感覺實際用途沒有那么大,但是對理解閉包對作用域鏈中的屬性的引用,這一點還是有作用的。 函數表達式和閉包 1. 函數聲明的一個重要特征是函數聲明提升 如: sayHi() function sayHi () { console.log(h...
摘要:說明此摘要筆記系列是我最近看高級程序設計第版隨手所記。其中,描述符對象的屬性必須是設置其中一個或多個值,可以修改對應的特性值。如支持的瀏覽器,可以取得指定屬性的描述符。 說明: 此摘要筆記系列是我最近看《JavaScript高級程序設計(第3版)》隨手所記。里面分條列舉了一些我認為重要的、需要記下的、對我有幫助的點,是按照我看的順序來的。摘要筆記本身沒有系統性,沒有全面性可言,寫在這里...
閱讀 1776·2021-11-11 11:02
閱讀 1679·2021-09-22 15:55
閱讀 2484·2021-09-22 15:18
閱讀 3488·2019-08-29 11:26
閱讀 3743·2019-08-26 13:43
閱讀 2646·2019-08-26 13:32
閱讀 897·2019-08-26 10:55
閱讀 965·2019-08-26 10:27