JavaScript 繼承 封裝 多型實現及原理詳解
面向物件的三大特性
封裝
所謂封裝,也就是把客觀事物封裝成抽象的類,並且類可以把自己的資料和方法只讓可信的類或者物件操作,對不可信的進行資訊隱藏。封裝是面向物件的特徵之一,是物件和類概念的主要特性。 簡單的說,一個類就是一個封裝了資料以及操作這些資料的程式碼的邏輯實體。在一個物件內部,某些程式碼或某些資料可以是私有的,不能被外界訪問。通過這種方式,物件對內部資料提供了不同級別的保護,以防止程式中無關的部分意外的改變或錯誤的使用了物件的私有部分。
我們在vue專案中使用混入將公有程式碼提出來,混入到每個需要的元件中,讓我們的程式碼更簡潔
我們也可以將一些公用的方法,工具進行封裝,達到程式碼複用,讓我們的程式碼更簡潔
繼承
所謂繼承是指可以讓某個型別的物件獲得另一個型別的物件的屬性的方法。它支援按級分類的概念。繼承是指這樣一種能力:它可以使用現有類的所有功能,並在無需重新編寫原來的類的情況下對這些功能進行擴充套件。 通過繼承建立的新類稱為“子類”或“派生類”,被繼承的類稱為“基類”、“父類”或“超類”。繼承的過程,就是從一般到特殊的過程。要實現繼承,可以通過“繼承”(Inheritance)和“組合”(Composition)來實現。繼承概念的實現方式有二類:實現繼承與介面繼承。實現繼承是指直接使用基類的屬性和方法而無需額外編碼的能力;介面繼承是指僅使用屬性和方法的名稱、但是子類必須提供實現的能力;
實現繼承的幾種方式
call繼承
function a (){ this.c=111 } function b(){ a.call(this) } let d = new b() console.log(d.c) // 111
上面程式碼相當於函式b繼承了函式a的私有屬性,通過改變父類的this實現繼承
原型繼承
function a (){ this.c=111 } a.prototype.getName=function(){ return '你好' } function b(){ // a.call(this) } b.prototype=new a() b.constructor=b let d = new b() console.log(d.c) // 111 console.log(d.getName()) // 你好
原型繼承通過將父類的例項賦值給子類的原型,這種方法子類可以繼承父類的私有方法也可以繼承父類的私有方法
寄生組合繼承
function a (){ this.c=111 } a.prototype.getName=function(){ return '你好' } function b(){ a.call(this) } b.prototype=Object.create(a.prototype) let d = new b() console.log(d.c) // 111 console.log(d.getName()) // 你好
寄生組合繼承就是使用call繼承改變this,實現私有繼承私有,使用object.create實現公有繼承公有
es6 extends 繼承
class parent{ constructor(){ this.a=1 } name(){ return 2 } } class child extends parent{ } let A = new child() console.log(A.a) // 1 console.log(A.name()) // 2
這裡通過關鍵字extends實現子類繼承父類的私有和公有,這裡需要注意如果子類裡面寫了constructor,就必須寫super否則會報錯
class parent{ constructor(){ this.a=1 } name(){ return 2 } } class child extends parent{ constructor(){ // 這裡不寫super會報錯,報錯資訊如下 } } // ncaught ReferenceError: Must call super constructor in derived class before accessing 'this' or returning from derived constructor at new child
多型
所謂多型就是指一個類例項的相同方法在不同情形有不同表現形式。多型機制使具有不同內部結構的物件可以共享相同的外部介面。這意味著,雖然針對不同物件的具體操作不同,但通過一個公共的類,它們(那些操作)可以通過相同的方式予以呼叫。
過載
函式過載是指在同一作用域內,可以有一組具有相同函式名,不同引數列表的函式,這組函式被稱為過載函式。過載函式通常用來命名一組功能相似的函式,這樣做減少了函式名的數量,避免了名字空間的汙染,對於程式的可讀性有很大的好處。
- js通過函式名找到對應的函式物件
- 然後根據函式按照定義時的引數,和表示式引數列表按順序匹配,多餘的引數捨去,不夠的引數按undefined處理
- 然後執行函式程式碼。
// 可以跟據arguments個數實現過載 function add() { var sum = 0 ; for ( var i = 0 ; i < arguments.length; i ++ ) { sum += arguments[i]; } return sum; } console.log(add()) // 0 console.log(add(1,2)) // 3 console.log(add(1,2,3)) // 6
重寫
“例項中的指標僅指向原型,而不是指向建構函式”。
“重寫原型物件切斷了現有原型與任何之前已經存在的物件例項之間的關係;它們引用的仍然是最初的原型”。
var parent = function(name,age){ this.name = name; this.age = age; } parent.prototype.showProper = function(){ console.log(this.name+":"+this.age); } var child = function(name,age){ parent.call(this,name,age); } // inheritance child.prototype = Object.create(parent.prototype); // child.prototype = new parent(); child.prototype.constructor = child; // rewrite function child.prototype.showProper = function(){ console.log('I am '+this.name+":"+this.age); } var obj = new child('wozien','22'); obj.showProper();
上面這段程式碼通過使用寄生組合繼承,實現子類私有繼承父類私有,子類公有繼承父類公有,達到重寫父類的showProper
面向物件的5大原則
- 單一職責原則
- 開放封閉原則
- 替換原則
- 依賴原則
- 介面分離原則
單一職責原則
單一職責原則就是我們說的一個方法只做一件事,就比如現在的專案結構,就遵循了單一職責原則
開放封閉原則
開放封閉原則就是對修改關閉,對擴充套件開放
class a { add(){ return 11 } } class b extends a{ } let c = new b() console.log(c.add()) // 111
我們可以使用extends繼承父類,可以再b裡面修改add函式,實現對修改關閉,對擴充套件開放
以上就是本文的全部內容,希望對大家的學習有所幫助
您可能感興趣的文章:
- JavaScript使用prototype原型實現的封裝繼承多型示例
- JavaScript實現多型和繼承的封裝操作示例
- Javascript基於物件三大特性(封裝性、繼承性、多型性)
- 不錯的一篇關於javascript-prototype繼承
- js封裝成外掛的步驟方法
- javascript 面向物件程式設計基礎 多型
文章同步釋出: https://www.geek-share.com/detai