淺析JS中的原型,原型鏈和繼承
經(jīng)典模式和圣杯模式區(qū)別
經(jīng)典模式和圣杯模式都是用于解決構(gòu)造函數(shù)繼承和原型繼承的問題,但它們在實現(xiàn)繼承的方式上有所不同。
經(jīng)典模式是通過將子類的原型對象設(shè)置為父類的實例來實現(xiàn)繼承,然后將子類的構(gòu)造函數(shù)設(shè)置為子類本身。這樣子類既可以繼承父類的屬性,也可以繼承父類原型對象的方法。但是經(jīng)典模式存在一個問題,就是每次創(chuàng)建子類的實例時都會調(diào)用一次父類的構(gòu)造函數(shù),導(dǎo)致父類的屬性被重復(fù)初始化。
圣杯模式是在經(jīng)典模式的基礎(chǔ)上進行了改進,通過使用一個中間函數(shù)來實現(xiàn)繼承。這個中間函數(shù)將父類的原型對象賦值給一個臨時的構(gòu)造函數(shù),并將子類的原型對象設(shè)置為這個臨時構(gòu)造函數(shù)的實例。這樣子類既可以繼承父類原型對象的方法,又不會重復(fù)調(diào)用父類的構(gòu)造函數(shù)。此外,圣杯模式還通過將子類的原型對象的constructor屬性設(shè)置為子類本身來修復(fù)原型鏈斷裂的問題。
下面是經(jīng)典模式和圣杯模式的代碼示例:
經(jīng)典模式:
function Parent(name) { this.name = name; } Parent.prototype.sayHello = function() { console.log('Hello, I am ' + this.name); } function Child(name) { Parent.call(this, name); } Child.prototype = new Parent(); Child.prototype.constructor = Child;
圣杯模式:
function Parent(name) { this.name = name; } Parent.prototype.sayHello = function() { console.log('Hello, I am ' + this.name); } function inherit(C, P) { function F() {} F.prototype = P.prototype; C.prototype = new F(); C.prototype.constructor = C; } function Child(name) { Parent.call(this, name); } inherit(Child, Parent);
經(jīng)典模式和圣杯模式都是用于實現(xiàn)繼承的方式,但圣杯模式在解決經(jīng)典模式中的問題上更加優(yōu)化和完善。
圣杯模式
1. 原型
原型是JavaScript中用來實現(xiàn)對象之間繼承關(guān)系的概念。每個JavaScript對象都有一個原型對象,它是一個普通的對象,包含了對象的屬性和方法。當我們訪問一個對象的屬性或方法時,如果對象本身不存在該屬性或方法,JavaScript會自動去原型對象中查找。
2. 原型鏈
原型鏈是一種通過原型對象來實現(xiàn)對象之間繼承關(guān)系的機制。每個對象都有一個原型對象,通過原型鏈,一個對象可以訪問其原型對象的屬性和方法。原型鏈是由一系列原型對象組成的,當我們訪問一個對象的屬性或方法時,JavaScript會自動沿著原型鏈向上查找,直到找到該屬性或方法或者到達原型鏈的頂端。
3. 繼承
繼承是一種面向?qū)ο缶幊讨械闹匾拍?,它允許我們創(chuàng)建一個新的對象,并從一個已有的對象中繼承屬性和方法。通過繼承,我們可以避免重復(fù)編寫相同的代碼,提高代碼的復(fù)用性和可維護性。
相應(yīng)的代碼示例
a. 原型鏈繼承
原型鏈繼承是一種簡單的繼承方式,它通過將子類的原型對象設(shè)置為父類的實例來實現(xiàn)繼承。這樣子類就可以訪問父類原型對象的屬性和方法。
function Parent() { this.name = 'Parent'; } Parent.prototype.sayHello = function() { console.log('Hello, I am ' + this.name); } function Child() { this.name = 'Child'; } Child.prototype = new Parent(); var child = new Child(); child.sayHello(); // 輸出: Hello, I am Child
在這個例子中,我們定義了一個父類Parent和一個子類Child。在子類中,我們將其原型對象設(shè)置為父類的實例,這樣子類就可以繼承父類的屬性和方法。在子類的實例中,我們可以調(diào)用父類原型對象的方法。
b. 構(gòu)造函數(shù)繼承
構(gòu)造函數(shù)繼承是一種通過調(diào)用父類的構(gòu)造函數(shù)來實現(xiàn)繼承的方式。子類通過調(diào)用父類的構(gòu)造函數(shù)來繼承父類的屬性,并在子類的構(gòu)造函數(shù)中使用call方法來調(diào)用父類的構(gòu)造函數(shù)。
function Parent(name) { this.name = name; } function Child(name) { Parent.call(this, name); } var child = new Child('Child'); console.log(child.name); // 輸出: Child
在這個例子中,我們定義了一個父類Parent和一個子類Child。在子類的構(gòu)造函數(shù)中,我們使用call方法調(diào)用父類的構(gòu)造函數(shù),并將子類的實例作為this參數(shù)傳遞給父類的構(gòu)造函數(shù)。這樣子類就可以繼承父類的屬性。
c. 組合繼承
組合繼承是一種通過同時使用原型鏈繼承和構(gòu)造函數(shù)繼承來實現(xiàn)繼承的方式。它通過將子類的原型對象設(shè)置為父類的實例,并在子類的構(gòu)造函數(shù)中調(diào)用父類的構(gòu)造函數(shù)來實現(xiàn)繼承。
function Parent(name) { this.name = name; } Parent.prototype.sayHello = function() { console.log('Hello, I am ' + this.name); } function Child(name) { Parent.call(this, name); } Child.prototype = new Parent(); Child.prototype.constructor = Child; var child = new Child('Child'); child.sayHello(); // 輸出: Hello, I am Child
在這個例子中,我們定義了一個父類Parent和一個子類Child。在子類的構(gòu)造函數(shù)中,我們使用call方法調(diào)用父類的構(gòu)造函數(shù),并將子類的實例作為this參數(shù)傳遞給父類的構(gòu)造函數(shù)。然后,我們將子類的原型對象設(shè)置為父類的實例,并將子類的構(gòu)造函數(shù)設(shè)置為子類本身。這樣子類既可以繼承父類的屬性,也可以繼承父類原型對象的方法。
詳細代碼說明
a. 原型鏈繼承
function Parent() { this.name = 'Parent'; } Parent.prototype.sayHello = function() { console.log('Hello, I am ' + this.name); } function Child() { this.name = 'Child'; } Child.prototype = Object.create(Parent.prototype); Child.prototype.constructor = Child; var child = new Child(); child.sayHello(); // 輸出: Hello, I am Child
在這個例子中,我們使用Object.create()方法將子類的原型對象設(shè)置為父類的原型對象的一個副本。這樣子類就可以繼承父類原型對象的屬性和方法。
b. 構(gòu)造函數(shù)繼承
function Parent(name) { this.name = name; } Parent.prototype.sayHello = function() { console.log('Hello, I am ' + this.name); } function Child(name) { Parent.call(this, name); } var child = new Child('Child'); child.sayHello(); // 輸出: TypeError: child.sayHello is not a function
在這個例子中,我們通過調(diào)用父類的構(gòu)造函數(shù)來繼承父類的屬性,但是子類無法繼承父類原型對象的方法。
c. 組合繼承
在這個例子中,我們使用Object.create()方法將子類的原型對象設(shè)置為父類的原型對象的一個副本,并將子類的構(gòu)造函數(shù)設(shè)置為子類本身。這樣子類既可以繼承父類的屬性,也可以繼承父類原型對象的方法。
function Parent(name) { this.name = name; } Parent.prototype.sayHello = function() { console.log('Hello, I am ' + this.name); } function Child(name) { Parent.call(this, name); } Child.prototype = Object.create(Parent.prototype); Child.prototype.constructor = Child; var child = new Child('Child'); child.sayHello(); // 輸出: Hello, I am Child
到此這篇關(guān)于淺析JS中的原型,原型鏈和繼承的文章就介紹到這了,更多相關(guān)JS原型 原型鏈 繼承內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
unicloud云開發(fā)進階獲取首頁列表數(shù)據(jù)示例詳解
這篇文章主要為大家介紹了unicloud云開發(fā)進階獲取首頁列表數(shù)據(jù)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-03-03js中方法重載如何實現(xiàn)?以及函數(shù)的參數(shù)問題
js中沒有辦法直接實現(xiàn)方法重載,但每一個函數(shù)都有一個特殊的參數(shù)arguments,利用它可以實現(xiàn)方法的重載,具體示例如下2013-08-08