摘要:繼承可以使得子類具有父類別的各種屬性和方法。繼承是類與類之間的關(guān)系。繼承的實質(zhì)就是兩次的原型搜索,像是實例屬性而不是繼承,才是繼承。更多用法見繼承。
前言
面試中最常會問到的問題:什么是繼承?如何分別用 ES5 和 ES6 實現(xiàn)?想要學習繼承,必須先學好原型與原型鏈,如果此部分還不清楚,請先學習此部分再來閱讀本文,可參考我的文章JS之原型與原型鏈或瀏覽其他相關(guān)的學習網(wǎng)站。
定義繼承到底是什么?維基百科是這樣說的:繼承(英語:inheritance)是面向?qū)ο筌浖夹g(shù)當中的一個概念。如果一個類別B“繼承自”另一個類別A,就把這個B稱為“A的子類”,而把A稱為“B的父類別”也可以稱“A是B的超類”。繼承可以使得子類具有父類別的各種屬性和方法。詳情點我。繼承是類與類之間的關(guān)系。
var a = new Array() a.valueOf()
上面的a為什么可以使用valueOf方法?實際上是它是通過了原型鏈的搜索最終在Object.prototype里拿到了valueOf方法。
繼承的實質(zhì)就是兩次的原型搜索,像a.push()是實例屬性而不是繼承,a.valueOf()才是繼承。
自己實現(xiàn)原型鏈繼承上面的結(jié)構(gòu)就是我們要實現(xiàn)的,其中子類的Human和父類Object之間的繼承是自帶的,我們需要在Human后再加一個Man讓person繼承Human的屬性和方法。
ES5實現(xiàn)繼承1.把Human的自身的屬性直接寫在Man上:Human.call(this, name)
function Human(name) { // 創(chuàng)建一個 Human 構(gòu)造函數(shù) this.name = name } Human.prototype.run = function () { console.log("I can run") } function Man(name) { // 創(chuàng)建一個 Man 構(gòu)造函數(shù) Human.call(this, name) this.gender = "男" } Man.prototype.fight = function () { console.log("I can fight") } var dong = new Man("dong")
以上代碼實現(xiàn)了上圖的效果,但是我們并沒有把Human也聯(lián)系起來,我們想做的就是在第一第二層中加上Human這一層,即形成dong → Man → Human → Object 鏈式,也就是實現(xiàn)下圖:
2.改變Man.prototype.__proto__的指向
所以我們只要把Man.prototype.__proto__的指向由Object.prototype變?yōu)橹赶騂uman.prototype即可,此時你可能覺得簡單的加上一句Man.prototype.__proto__ = Human.prototype就可以改變它的指向,確實這種寫法成功的改變了Man.prototype.__proto__的指向,但由于IE不支持這樣的寫法,所以我們要把代碼修改為:
var f = function(){} f.prototype = Human.prototype Man.prototype = new f()
3.代碼總結(jié)
function Human(name) { // 創(chuàng)建一個 Human 構(gòu)造函數(shù) this.name = name } Human.prototype.run = function () { console.log("I can run") } function Man(name) { // 創(chuàng)建一個 Man 構(gòu)造函數(shù) Human.call(this, name) this.gender = "男" } var f = function(){} f.prototype = Human.prototype Man.prototype = new f() //這里需要注意的問題是被修改的原型鏈的屬性必須修改完后才能聲明 //也就是這句話必須得寫在fight 前,否則原型鏈一修改它里面的屬性和方法就沒了 Man.prototype.fight = function () { console.log("I can fight") } var dong = new Man("dong")ES6實現(xiàn)繼承(class 和 extends)
class Human { constructor(name) { this.name = name } run() { console.log("I can run") } } class Man extends Human { //Man.prototype.__proto__ = Human.prototype constructor(name) { super(name) //Human.call(this,name) this.gender = "男" } fight(){ console.log("I can fight") } }
ES6中類的寫法:自身屬性name寫在constructor里面,共有屬性(原型鏈上的)run直接和他并列著寫,而繼承的寫法:class后面是子類然后extends你的父類。更多用法見ES6繼承。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/106598.html
摘要:在中必須調(diào)用方法,因為子類沒有自己的對象,而是繼承父類的對象,然后對其進行加工而就代表了父類的構(gòu)造函數(shù)。雖然代表了父類的構(gòu)造函數(shù),但是返回的是子類的實例,即內(nèi)部的指的是,因此在這里相當于。要求,子類的構(gòu)造函數(shù)必須執(zhí)行一次函數(shù),否則會報錯。 1.class ES6 提供了更接近傳統(tǒng)語言的寫法,引入了 Class(類)這個概念,作為對象的模板。通過class關(guān)鍵字,可以定義類?;旧?,ES...
摘要:在中必須調(diào)用方法,因為子類沒有自己的對象,而是繼承父類的對象,然后對其進行加工而就代表了父類的構(gòu)造函數(shù)。雖然代表了父類的構(gòu)造函數(shù),但是返回的是子類的實例,即內(nèi)部的指的是,因此在這里相當于。要求,子類的構(gòu)造函數(shù)必須執(zhí)行一次函數(shù),否則會報錯。 1.class ES6 提供了更接近傳統(tǒng)語言的寫法,引入了 Class(類)這個概念,作為對象的模板。通過class關(guān)鍵字,可以定義類?;旧?,ES...
摘要:返回布爾值,表示參數(shù)字符串是否在源字符串的頭部。參考語法返回一個布爾值與的全等操作符比較兼容環(huán)境把對象的值復(fù)制到另一個對象里淺拷貝定義方法用于將所有可枚舉的屬性的值從一個或多個源對象復(fù)制到目標對象。語法要設(shè)置其原型的對象。 一步一步似爪牙。 前言 學習es6之前我們可能并不知道es6相比es5差距在哪, 但是這并不妨礙我們站在巨人的肩膀上; 程序員就是要樂于嘗鮮; 學習es6最終目的是...
摘要:生成的類的原型會被自動調(diào)整,而你還能調(diào)用方法來訪問基類的構(gòu)造器。唯一能避免調(diào)用的辦法,是從類構(gòu)造器中返回一個對象。 起源 JS 從創(chuàng)建之初就不支持類,也沒有把類繼承作為定義相似對象以及關(guān)聯(lián)對象的主要方式,這讓不少開發(fā)者感到困惑。而從 ES1 誕生之前直到ES5 時期,很多庫都創(chuàng)建了一些工具,讓 JS 顯得貌似能支持類。盡管一些 JS 開發(fā)者強烈認為這門語言不需要類,但為處理類而創(chuàng)建的代...
閱讀 1146·2019-08-30 12:44
閱讀 652·2019-08-29 13:03
閱讀 2562·2019-08-28 18:15
閱讀 2431·2019-08-26 10:41
閱讀 3091·2019-08-26 10:28
閱讀 3039·2019-08-23 16:54
閱讀 1991·2019-08-23 15:16
閱讀 817·2019-08-23 14:55