原型模式和基于原型繼承的javascript系統(tǒng)

導(dǎo)言:

在以類(lèi)為中心的面向?qū)ο缶幊陶Z(yǔ)言中,類(lèi)和對(duì)象的關(guān)系可以想象成鑄模和鑄件的關(guān)系汽烦,對(duì)象總是總類(lèi)中創(chuàng)建而來(lái)涛菠。而在原型編程思想中,類(lèi)并不是必須的刹缝,對(duì)象也不必要從類(lèi)中創(chuàng)建而來(lái)碗暗,一個(gè)對(duì)象是克隆另外一個(gè)對(duì)象創(chuàng)建而來(lái)的,原型模式不僅僅是一種設(shè)計(jì)模式梢夯,也是一種編程范式

1.使用克隆的原型模式

如果要使用原型模式言疗,那么我們只需要負(fù)責(zé)調(diào)用克隆的方法,便能完成克隆對(duì)象的功能颂砸。原型模式的實(shí)現(xiàn)關(guān)鍵是語(yǔ)言本身是否提供了克隆方法噪奄,ECMAscript5提供了一個(gè)方法可以用來(lái)克隆對(duì)象這個(gè)方法是Object.creat().代碼如下:

var Plane = function () {

? ? ? this.blood = 100;

? ? ? this.attracklevel=1;

? ? ? this.defenselevel=0.5

}

var plane = new Plane()

plane.blood = 500;

plane.attracklevel=10

var clonePlane = Object.create(plane) ?// 克隆對(duì)象

clonePlane.blood // ?500

以上代碼是通過(guò)使用Object.cteate()來(lái)克隆一個(gè)對(duì)象的副本,但是有些瀏覽器并不支持Object.create()方法人乓,那么這是我們?yōu)榱俗黾嫒菘梢赃M(jìn)行這樣的處理勤篮,代碼如下:

Object.create = Object.create || function (obj) { // obj是要被克隆的對(duì)象

? ? // 通過(guò)構(gòu)造函數(shù)的方式來(lái)克隆

? ? var F = function () {}

? ? F.prototype = obj;

? ? return new F()

}

在瀏覽器控制臺(tái)運(yùn)行這段代碼:

var a = {name: 'steve'}

var b = Object.create(a) // 得到對(duì)象a的一個(gè)副本,但是對(duì)象a和對(duì)象b的地址引用肯定是不相等的

console.log(b) ?// 得到的是steve

2.克隆是創(chuàng)建對(duì)象的手段

????上面的代碼中我們知道了如何通過(guò)原型模式來(lái)克隆出一個(gè)對(duì)象色罚,但是原型模式的真正目的并不是創(chuàng)建出一個(gè)一模一樣的對(duì)象碰缔,而是提供了便捷的方式去創(chuàng)建某個(gè)類(lèi)型的對(duì)象,克隆只是創(chuàng)建對(duì)象的過(guò)程和手段戳护。

? ? 我們?cè)谶M(jìn)行編程的時(shí)候有一個(gè)非常重要的原則金抡,那就是依賴導(dǎo)致原則,那么什么是依賴倒置原則了腌且?所謂的依賴倒置就是說(shuō)創(chuàng)建對(duì)象的時(shí)候避免依賴具體的類(lèi)型梗肝,就像java等靜態(tài)類(lèi)型的語(yǔ)言進(jìn)行編程的時(shí)候,類(lèi)型之間的解耦是非常重要的铺董,通過(guò)new XXX的方式創(chuàng)建對(duì)象顯得很僵硬巫击,因此就有了工廠方法和抽象工廠模式方法來(lái)幫助我們解決這個(gè)問(wèn)題,但是這兩種方法帶來(lái)的結(jié)果是最終會(huì)出現(xiàn)很多平行的工廠類(lèi)精续,這就會(huì)導(dǎo)致額外的代碼坝锰,如有一個(gè)創(chuàng)建狗的工廠:

function dogFactory (name,age) {

? ? var obj = new Object();

? ? obj.name = name;

? ? obj.age = age

? ? return obj

}

有一個(gè)創(chuàng)建貓的工廠:

function catFactory (name,age) {

var obj = new Object();

obj.name = name;

obj.age = age

return obj

}

很明顯創(chuàng)建狗的工廠和創(chuàng)建貓的工廠的內(nèi)部的代碼是一模一樣的,只是工廠名字變了驻右,由此可見(jiàn)這兩個(gè)平行的工廠額外的增加了代碼量什黑,這顯然是不期望的。在js中堪夭,原型模式為我們提供了另外一種創(chuàng)建對(duì)象的方式愕把,就是通過(guò)克隆對(duì)象拣凹,而不是像上面一樣從一個(gè)工廠類(lèi)中創(chuàng)建對(duì)象。通過(guò)克隆對(duì)象我們就不在關(guān)心對(duì)象的具體類(lèi)型是什么了恨豁,因此通過(guò)原型模式創(chuàng)建對(duì)象就會(huì)顯得非常容易嚣镜,也不存在類(lèi)型耦合的問(wèn)題。其實(shí)javascript的對(duì)象系統(tǒng)就是通過(guò)原型模式來(lái)搭建的橘蜜,所有的對(duì)象都是從某個(gè)對(duì)象上克隆而來(lái)的菊匿。既然整個(gè)對(duì)象系統(tǒng)是基于原型搭建的,那么這種原型模式也一定遵循一些編程規(guī)范计福,接下來(lái)了解一下原型編程規(guī)范的基本規(guī)則

3原型編程規(guī)范基本規(guī)則

1.所有的數(shù)據(jù)都是對(duì)象跌捆,例如前面的var F = function(){},這個(gè)函數(shù)不僅僅是一個(gè)函數(shù)象颖,他也是一個(gè)對(duì)象佩厚,其實(shí)函數(shù)在js中就是一等公民,也就是說(shuō)函數(shù)就是對(duì)象说订,具體一點(diǎn)的解釋如下:

javascript在設(shè)計(jì)的時(shí)候抄瓦,模仿java引入了兩套類(lèi)型機(jī)制,基本類(lèi)型和引用類(lèi)型陶冷,基本類(lèi)型包括

undefined,number,bool,string,function,object,但是這并不是一個(gè)好的想法钙姊,按照javascript設(shè)計(jì)者本意的想法是出來(lái)undefined,其余的都是對(duì)象,為了實(shí)現(xiàn)這一目標(biāo)埂伦,number,bool,string這幾種基本類(lèi)型數(shù)據(jù)可以通過(guò)包裝類(lèi)型的方式變成對(duì)象類(lèi)型數(shù)據(jù)來(lái)處理煞额。什么意思了?有點(diǎn)不明白沾谜,function 立镶,object是對(duì)象是可以接受的,但是number,bool,string 不就是一個(gè)數(shù)字类早,一個(gè)布爾值,一個(gè)字符串嗎嗜逻,怎么能成為對(duì)象了涩僻,這顯然很矛盾,但時(shí)當(dāng)我們知道了包裝類(lèi)型的時(shí)候問(wèn)題就會(huì)立馬得到解決栈顷,所謂的包裝類(lèi)型就是javascript提供了Numer,Bool,String這樣的構(gòu)造函數(shù)逆日,通過(guò)將上面幾個(gè)基本類(lèi)型的首字母換成大寫(xiě)字母就得到了對(duì)應(yīng)的包裝類(lèi)型,如通過(guò)var s = String('abc')可以將字符創(chuàng)abc轉(zhuǎn)換為對(duì)象類(lèi)型來(lái)處理,

事實(shí)上萄凤,我們并不能說(shuō)在javascript中所有的數(shù)據(jù)都是對(duì)象室抽,但可以說(shuō)絕大部分是對(duì)象,那么我們相信在js中一定會(huì)有一個(gè)根對(duì)象存在靡努,其實(shí)這個(gè)根對(duì)象就是Object.prototype(是Object構(gòu)造器的原型)坪圾,這個(gè)對(duì)象是一個(gè)空的對(duì)象晓折,其實(shí)我們遇到的每一個(gè)對(duì)象最終都是從Object.prototype克隆而來(lái)的。如:

創(chuàng)建兩個(gè)對(duì)象obj1和obj2兽泄,

var obj1 = {} ;var obj2 = new Object();利用es5提供的getPrototypeOf()來(lái)查看這兩個(gè)對(duì)象的原型

console.log(Object.getPrototypeOf(obj1) === Object.prototype) // true

console.log(Object.getPrototypeOf(obj2) === Object.prototype) ?// true

2.要得到一個(gè)對(duì)象不是通過(guò)實(shí)例化類(lèi)漓概,而是找到一個(gè)對(duì)象作為原型并且克隆他。

如:找到一個(gè)對(duì)象a = {name: 'steve'},將啊作為F的原型來(lái)克隆a

var creatObj = function () {

? ? var F = function () {}

? ? F.prototype = a

? ? return new F() // 得到一個(gè)a的副本,到這一步我們應(yīng)該很困惑病梢,為什么沒(méi)有類(lèi)的概念胃珍,而在這里卻用了new,其實(shí)這里的F比不是類(lèi)蜓陌,而是一個(gè)構(gòu)造函數(shù)觅彰,這里使用new運(yùn)算符來(lái)創(chuàng)建兌現(xiàn)的過(guò)程實(shí)際上也只是先克隆Object.prototype對(duì)象,然后在進(jìn)行一些其他額外的操作

}

可已通過(guò)下面的代碼來(lái)理解new運(yùn)算的過(guò)程

// 創(chuàng)建一個(gè)構(gòu)造函數(shù)Person

function Person (name) {

? ? ? ? this.name = name

}

// 在構(gòu)造函數(shù)的原型對(duì)象上掛在一個(gè)方法

Person.prototype.getName = function () {

? return this.name

}

// 創(chuàng)建一個(gè)克隆兌現(xiàn)的工廠

var objFactory = function () {

? ? // 創(chuàng)建一個(gè)對(duì)象钮热,將其加工處理后最終返回出去

? ? var obj ?= ?new Object()

? ? Consructor = [].shift.call(arguments) // 拿到第一個(gè)參數(shù)填抬,這里是一個(gè)構(gòu)造函數(shù),shift會(huì)改變?cè)瓟?shù)組

? ? obj._proto_ = Constructor.prototype ?// 讓obj指向正確的原型

? ? var ret = Constructor.apply(obj,arguments) // 借用外部傳入的構(gòu)造器給obj設(shè)置屬性霉旗,也就是我前面說(shuō)的對(duì)obj進(jìn)行加工處理

? ? return ?typeof ret === 'object' ? ret : obj // 既然是對(duì)象工廠痴奏,那么最終一定要返回出去一個(gè)對(duì)象

}

var a = objFactory(Person,'steve')

console.log(a.name) // steve

console.log(a.getName()) // steve

console.log(Object.getPrototypeOf(a) === Person.prototype)// true, 很明顯得到兌現(xiàn)的原型就是對(duì)應(yīng)構(gòu)造函數(shù)的原型

3.對(duì)象會(huì)記住他的原型: 所有的對(duì)象都有一個(gè)_proto_屬性,對(duì)象就是通過(guò)這個(gè)屬性來(lái)記住他的原型的

4.如果對(duì)象無(wú)法響應(yīng)某個(gè)請(qǐng)求厌秒,那么就會(huì)把請(qǐng)求委托給他的原型

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末读拆,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子鸵闪,更是在濱河造成了極大的恐慌檐晕,老刑警劉巖,帶你破解...
    沈念sama閱讀 210,914評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蚌讼,死亡現(xiàn)場(chǎng)離奇詭異辟灰,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)篡石,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,935評(píng)論 2 383
  • 文/潘曉璐 我一進(jìn)店門(mén)芥喇,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人凰萨,你說(shuō)我怎么就攤上這事继控。” “怎么了胖眷?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,531評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵武通,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我珊搀,道長(zhǎng)冶忱,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,309評(píng)論 1 282
  • 正文 為了忘掉前任境析,我火速辦了婚禮囚枪,結(jié)果婚禮上派诬,老公的妹妹穿的比我還像新娘。我一直安慰自己眶拉,他們只是感情好千埃,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,381評(píng)論 5 384
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著忆植,像睡著了一般放可。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上朝刊,一...
    開(kāi)封第一講書(shū)人閱讀 49,730評(píng)論 1 289
  • 那天耀里,我揣著相機(jī)與錄音,去河邊找鬼拾氓。 笑死冯挎,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的咙鞍。 我是一名探鬼主播房官,決...
    沈念sama閱讀 38,882評(píng)論 3 404
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼续滋!你這毒婦竟也來(lái)了翰守?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,643評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤疲酌,失蹤者是張志新(化名)和其女友劉穎蜡峰,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體朗恳,經(jīng)...
    沈念sama閱讀 44,095評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡湿颅,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,448評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了粥诫。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片油航。...
    茶點(diǎn)故事閱讀 38,566評(píng)論 1 339
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖怀浆,靈堂內(nèi)的尸體忽然破棺而出劝堪,到底是詐尸還是另有隱情,我是刑警寧澤揉稚,帶...
    沈念sama閱讀 34,253評(píng)論 4 328
  • 正文 年R本政府宣布,位于F島的核電站熬粗,受9級(jí)特大地震影響搀玖,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜驻呐,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,829評(píng)論 3 312
  • 文/蒙蒙 一灌诅、第九天 我趴在偏房一處隱蔽的房頂上張望芳来。 院中可真熱鬧,春花似錦猜拾、人聲如沸即舌。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,715評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)顽聂。三九已至,卻和暖如春盯仪,著一層夾襖步出監(jiān)牢的瞬間紊搪,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,945評(píng)論 1 264
  • 我被黑心中介騙來(lái)泰國(guó)打工全景, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留耀石,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,248評(píng)論 2 360
  • 正文 我出身青樓爸黄,卻偏偏與公主長(zhǎng)得像滞伟,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子炕贵,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,440評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容