摘要:類的實(shí)現(xiàn)轉(zhuǎn)換前轉(zhuǎn)換后可見的底層依然是構(gòu)造函數(shù)調(diào)用方法判斷當(dāng)前函數(shù)調(diào)用前是否有關(guān)鍵字。若構(gòu)造函數(shù)前面沒有則構(gòu)造函數(shù)的不會(huì)不出現(xiàn)在的原型鏈上,返回。典型的寄生繼承用父類構(gòu)造函數(shù)的創(chuàng)建一個(gè)空對(duì)象,并將這個(gè)對(duì)象指向子類構(gòu)造函數(shù)的。代表父類構(gòu)造函數(shù)。
在閱讀文章之前,您至少需要對(duì)JavaScript原型繼承有一定了解,如果覺得有所欠缺,可以先了解下我這篇文章:https://segmentfault.com/a/11...1.es6 class 使用
javascript使用的是原型式繼承,我們可以通過原型的特性實(shí)現(xiàn)類的繼承,
es6為我們提供了像面向?qū)ο罄^承一樣的語法糖。
class Parent { constructor(a){ this.filed1 = a; } filed2 = 2; func1 = function(){} } class Child extends Parent { constructor(a,b) { super(a); this.filed3 = b; } filed4 = 1; func2 = function(){} }
下面我們借助babel來探究es6類和繼承的實(shí)現(xiàn)原理。
1.類的實(shí)現(xiàn)轉(zhuǎn)換前:
class Parent { constructor(a){ this.filed1 = a; } filed2 = 2; func1 = function(){} }
轉(zhuǎn)換后:
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } var Parent = function Parent(a) { _classCallCheck(this, Parent); this.filed2 = 2; this.func1 = function () { }; this.filed1 = a; };
可見class的底層依然是構(gòu)造函數(shù):
1.調(diào)用_classCallCheck方法判斷當(dāng)前函數(shù)調(diào)用前是否有new關(guān)鍵字。
構(gòu)造函數(shù)執(zhí)行前有new關(guān)鍵字,會(huì)在構(gòu)造函數(shù)內(nèi)部創(chuàng)建一個(gè)空對(duì)象,將構(gòu)造函數(shù)的proptype指向這個(gè)空對(duì)象的_proto_,并將this指向這個(gè)空對(duì)象。如上,_classCallCheck中:this instanceof Parent 返回true。若構(gòu)造函數(shù)前面沒有new則構(gòu)造函數(shù)的proptype不會(huì)不出現(xiàn)在this的原型鏈上,返回false。
2.將class內(nèi)部的變量和函數(shù)賦給this。
3.執(zhí)行constuctor內(nèi)部的邏輯。
4.return this (構(gòu)造函數(shù)默認(rèn)在最后我們做了)。
2.繼承實(shí)現(xiàn)轉(zhuǎn)換前:
class Child extends Parent { constructor(a,b) { super(a); this.filed3 = b; } filed4 = 1; func2 = function(){} }
轉(zhuǎn)換后:
我們先看Child內(nèi)部的實(shí)現(xiàn),再看內(nèi)部調(diào)用的函數(shù)是怎么實(shí)現(xiàn)的:
var Child = function (_Parent) { _inherits(Child, _Parent); function Child(a, b) { _classCallCheck(this, Child); var _this = _possibleConstructorReturn(this, (Child.__proto__ || Object.getPrototypeOf(Child)).call(this, a)); _this.filed4 = 1; _this.func2 = function () {}; _this.filed3 = b; return _this; } return Child; }(Parent);
1.調(diào)用_inherits函數(shù)繼承父類的proptype。
_inherits內(nèi)部實(shí)現(xiàn):
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
(1) 校驗(yàn)父構(gòu)造函數(shù)。
(2) 典型的寄生繼承:用父類構(gòu)造函數(shù)的proptype創(chuàng)建一個(gè)空對(duì)象,并將這個(gè)對(duì)象指向子類構(gòu)造函數(shù)的proptype。
(3) 將父構(gòu)造函數(shù)指向子構(gòu)造函數(shù)的_proto_(這步是做什么的不太明確,感覺沒什么意義。)
2.用一個(gè)閉包保存父類引用,在閉包內(nèi)部做子類構(gòu)造邏輯。
3.new檢查。
4.用當(dāng)前this調(diào)用父類構(gòu)造函數(shù)。
var _this = _possibleConstructorReturn(this, (Child.__proto__ || Object.getPrototypeOf(Child)).call(this, a));
這里的Child.__proto__ || Object.getPrototypeOf(Child)實(shí)際上是父構(gòu)造函數(shù)(_inherits最后的操作),然后通過call將其調(diào)用方改為當(dāng)前this,并傳遞參數(shù)。(這里感覺可以直接用參數(shù)傳過來的Parent)
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn"t been initialised - super() hasn"t been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
校驗(yàn)this是否被初始化,super是否調(diào)用,并返回父類已經(jīng)賦值完的this。
5.將行子類class內(nèi)部的變量和函數(shù)賦給this。
6.執(zhí)行子類constuctor內(nèi)部的邏輯。
可見,es6實(shí)際上是為我們提供了一個(gè)“組合寄生繼承”的簡單寫法。
3. supersuper代表父類構(gòu)造函數(shù)。
super.fun1() 等同于 Parent.fun1() 或 Parent.prototype.fun1()。
super() 等同于Parent.prototype.construtor()
當(dāng)我們沒有寫子類構(gòu)造函數(shù)時(shí):
var Child = function (_Parent) { _inherits(Child, _Parent); function Child() { _classCallCheck(this, Child); return _possibleConstructorReturn(this, (Child.__proto__ || Object.getPrototypeOf(Child)).apply(this, arguments)); } return Child; }(Parent);
可見默認(rèn)的構(gòu)造函數(shù)中會(huì)主動(dòng)調(diào)用父類構(gòu)造函數(shù),并默認(rèn)把當(dāng)前constructor傳遞的參數(shù)傳給了父類。
所以當(dāng)我們聲明了constructor后必須主動(dòng)調(diào)用super(),否則無法調(diào)用父構(gòu)造函數(shù),無法完成繼承。
典型的例子就是Reatc的Component中,我們聲明constructor后必須調(diào)用super(props),因?yàn)楦割愐跇?gòu)造函數(shù)中對(duì)props做一些初始化操作。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/100829.html
摘要:使用新的易用的類定義,歸根結(jié)底也是要?jiǎng)?chuàng)建構(gòu)造函數(shù)和修改原型。首先,它把構(gòu)造函數(shù)當(dāng)成單獨(dú)的函數(shù)且包含類屬性集。該節(jié)點(diǎn)還儲(chǔ)存了指向父類的指針引用,該父類也并儲(chǔ)存了構(gòu)造函數(shù),屬性集和及父類引用,依次類推。 原文請(qǐng)查閱這里,略有刪減,本文采用知識(shí)共享署名 4.0 國際許可協(xié)議共享,BY Troland。 本系列持續(xù)更新中,Github 地址請(qǐng)查閱這里。 這是 JavaScript 工作原理的第...
摘要:使用新的易用的類定義,歸根結(jié)底也是要?jiǎng)?chuàng)建構(gòu)造函數(shù)和修改原型。首先,它把構(gòu)造函數(shù)當(dāng)成單獨(dú)的函數(shù)且包含類屬性集。該節(jié)點(diǎn)還儲(chǔ)存了指向父類的指針引用,該父類也并儲(chǔ)存了構(gòu)造函數(shù),屬性集和及父類引用,依次類推。 原文請(qǐng)查閱這里,略有刪減,本文采用知識(shí)共享署名 4.0 國際許可協(xié)議共享,BY Troland。 本系列持續(xù)更新中,Github 地址請(qǐng)查閱這里。 這是 JavaScript 工作原理的第...
摘要:下面是用實(shí)現(xiàn)轉(zhuǎn)成抽象語法樹如下還支持繼承以下是轉(zhuǎn)換結(jié)果最終的結(jié)果還是代碼,其中包含庫中的一些函數(shù)。可以使用新的易于使用的類定義,但是它仍然會(huì)創(chuàng)建構(gòu)造函數(shù)和分配原型。 這是專門探索 JavaScript 及其所構(gòu)建的組件的系列文章的第 15 篇。 想閱讀更多優(yōu)質(zhì)文章請(qǐng)猛戳GitHub博客,一年百來篇優(yōu)質(zhì)文章等著你! 如果你錯(cuò)過了前面的章節(jié),可以在這里找到它們: JavaScript 是...
摘要:首先有一個(gè)地方就是把指向了本身,這里有什么用呢,的時(shí)候是指向一個(gè)這樣的匿名函數(shù)。我們平時(shí)使用的和就是存在這個(gè)匿名函數(shù)里的。 理解es6的類和extends的原理,主要是先理解es5里面函數(shù)和對(duì)象間的關(guān)系(__proto__和prototype的知識(shí)) class c { constructor(){} //static f() {} a() {} } cl...
摘要:因?yàn)椴僮鞣麆?chuàng)建的對(duì)象都繼承自構(gòu)造函數(shù)的屬性。繼承的實(shí)現(xiàn)中常用的繼承方式是組合繼承,也就是通過構(gòu)造函數(shù)和原型鏈繼承同時(shí)來模擬繼承的實(shí)現(xiàn)。 原文發(fā)布在我的博客 我們都知道 JavaScript 是一門基于原型的語言。當(dāng)我們調(diào)用一個(gè)對(duì)象本身沒有的屬性時(shí),JavaScript 就會(huì)從對(duì)象的原型對(duì)象上去找該屬性,如果原型上也沒有該屬性,那就去找原型的原型,一直找原型鏈的末端也就是 Object....
閱讀 1323·2019-08-30 15:44
閱讀 2033·2019-08-30 13:49
閱讀 1664·2019-08-26 13:54
閱讀 3498·2019-08-26 10:20
閱讀 3288·2019-08-23 17:18
閱讀 3306·2019-08-23 17:05
閱讀 2139·2019-08-23 15:38
閱讀 1025·2019-08-23 14:35