我來(lái)重新學(xué)習(xí) javascript 的面向?qū)ο螅╬art 1)

很多job 的描述都說(shuō)要求精通 javascript 面向?qū)ο缶幊蹋歉鶕?jù)一般的套路蛀恩,寫(xiě)精通其實(shí)就是熟練,寫(xiě)熟練其實(shí)就是一般双谆,寫(xiě)一般其實(shí)就是懵逼!

image

雖然話(huà)說(shuō)如此顽馋,但是我們還是要熟練使用 javascript 面向?qū)ο缶幊痰模吘惯@是js社會(huì)高能人才的其中一個(gè)標(biāo)準(zhǔn)寸谜,這里我就用一個(gè)鮮活的例子來(lái)說(shuō)明和理解我們應(yīng)該如何使用javascript 面向?qū)ο蟮姆绞絹?lái)編程。

一、野蠻方式構(gòu)建對(duì)象

剛開(kāi)始最初地啰,我們創(chuàng)建對(duì)象的方式是這樣的:

// 。亏吝。。蔚鸥。每次都要寫(xiě)上面的一大段代碼,只是為了創(chuàng)建一個(gè) food
var food = new Object();
food.name = "蘋(píng)果";

food.sayName = function() {
  console.log("我是" + this.name);
};

但是這樣創(chuàng)建起來(lái)很麻煩止喷,寫(xiě)的代碼也是很長(zhǎng),如果要?jiǎng)?chuàng)建好多對(duì)象弹谁,例如我制造了10000個(gè)食物,就要寫(xiě)10000次這一大段代碼了预愤,所以后來(lái)聰明的工程師改為了這樣寫(xiě):

// 起碼比之前的少了幾行,也整潔了一些
var food = {
  name: "蘋(píng)果",
  sayName: function() {
    console.log("我是" + this.name);
  }
};

起碼代碼少了一些植康,但是還是沒(méi)辦法很好解決我要寫(xiě)100000段代碼的問(wèn)題,所以再后來(lái)的人們就開(kāi)始使用一些高級(jí)玩意來(lái)解決這個(gè)問(wèn)題销睁。

二、使用工廠模式構(gòu)建對(duì)象

通過(guò)抽象出創(chuàng)建具體對(duì)象的過(guò)程存崖,用函數(shù)來(lái)進(jìn)行封裝,換句話(huà)來(lái)說(shuō)来惧,就是抽象了一個(gè) food 的工廠,然后通過(guò)對(duì)這個(gè)工廠傳入不同的材料违寞,來(lái)生成不同的食物。

function createFood(name) {
  var o = new Object();
  o.name = name;
  o.sayName = function() {
    console.log("我是" + this.name);
  };
  return o;
}

var food1 = createFood("蘋(píng)果");
var food2 = createFood("蘋(píng)果");

這里可以看到food1趁曼,food2 就是這樣被制造出來(lái)的,然后只需要少量的代碼(預(yù)先定義好一個(gè)生產(chǎn)工廠函數(shù))挡闰,就可以完成大量的事情掰盘,徹底解決了問(wèn)題赞季,實(shí)現(xiàn)了多快好爽的新局面愧捕。但是用了一段時(shí)間之后申钩,隨之而來(lái)發(fā)現(xiàn)一個(gè)新問(wèn)題次绘,當(dāng)食物多起來(lái)的時(shí)候撒遣,老板貌似不知道哪些食物是屬于那些分類(lèi)的(假設(shè)老板是 zz),那怎么辦呢义黎?

// 都統(tǒng)一返回是[Function: Object],沒(méi)辦法用區(qū)分識(shí)別(賣(mài)個(gè)關(guān)子廉涕,你不用管那個(gè)constructor)
console.log(food1.constructor) // 返回[Function: Object]
console.log(food2.constructor) // 返回[Function: Object]

三、使用構(gòu)造函數(shù)模式來(lái)區(qū)分自己人

經(jīng)過(guò)一番智慧交流之后狐蜕,聰明的人們想出了一個(gè)方法宠纯,使用一個(gè)在對(duì)象里面的 constructor 函數(shù)來(lái)識(shí)別那些不一樣的對(duì)象馏鹤,類(lèi)似使用部門(mén)工牌來(lái)標(biāo)記這個(gè)人是是屬于哪個(gè)部門(mén)的娇哆。

function Food(name) {
  this.name = name;
  this.sayName = function() {
    console.log("我是" + this.name);
  };
}

var food1 = new Food("蘋(píng)果");
var food2 = new Food("蘋(píng)果");

// 假設(shè)這里有一個(gè)其他的食物湃累,可能是冒充的
var food3 = new otherFood("蘋(píng)果");

因?yàn)橐獙?shí)現(xiàn)類(lèi)似工牌的方式來(lái)識(shí)別碍讨,所以在創(chuàng)建food的工廠里做一些調(diào)整:

  • 沒(méi)有顯式的創(chuàng)建對(duì)象,例如:var o = new Object();
  • 直接將屬性和方法付給了 this 對(duì)象
  • 沒(méi)有 return 語(yǔ)句
  • 函數(shù)使用了大寫(xiě)字母開(kāi)頭(這里只是為了區(qū)分這個(gè)函數(shù)的特別勃黍,按照慣例,大寫(xiě)字母開(kāi)頭的覆获,一般都是 class 或者構(gòu)造函數(shù))
  • 使用了 new 來(lái)創(chuàng)建`Food``對(duì)象

做了以上的改變之后,整個(gè)創(chuàng)建對(duì)象的模式被改變了:

  1. 首先定義了一個(gè) Food 的構(gòu)造函數(shù)(其實(shí)就是之前的工廠函數(shù)createFood弄息,但是現(xiàn)在升級(jí)了)
  2. 通過(guò) new 來(lái)創(chuàng)建一個(gè)對(duì)象(現(xiàn)在的 Food 用 new 來(lái)先創(chuàng)建)
  3. 將構(gòu)造函數(shù)的作用域賦值給新對(duì)象,將this指向這個(gè)新對(duì)象(將升級(jí)版的工廠送給這個(gè)用 new 創(chuàng)建的 food)
  4. 執(zhí)行構(gòu)造函數(shù)的中的代碼(升級(jí)版的工廠會(huì)自動(dòng)將里面的零件和機(jī)器放到新的 Food 上摹量,相當(dāng)于組裝放在了食物本身 身上)
  5. 不需要主動(dòng) return馒胆,自動(dòng)返回新對(duì)象(升級(jí)版的工廠會(huì)自動(dòng)返回構(gòu)造好的 food 對(duì)象)

通過(guò)這種方式,我們制造出來(lái)的食物都會(huì)有一個(gè) constructor 為 Food 的標(biāo)記來(lái)標(biāo)識(shí)祝迂,如果看到不是的話(huà),那肯定就不是我們制造的型雳。

console.log(food1.constructor) // 返回[Function: Food]
console.log(food2.constructor) // 返回[Function: Food]
console.log(food3.constructor) // 返回[Function: OtherFood]

// 檢驗(yàn)的方式有兩種
console.log(food1.constructor == Food) // 返回 true
console.log(food2.constructor == Food) // 返回 true
console.log(food3.constructor == Food) // 返回 false ,這個(gè)不是我們制造的食物

console.log(food1 instanceof Food) // 返回 true
console.log(food3 instanceof Food) // 返回 false四啰,這個(gè)不是我們制造的食物

可以看到,使用了新技術(shù)(constructor模式技術(shù))之后柑晒,在沒(méi)有增加工作量的情況下,解決了令人頭痛的問(wèn)題匙赞,簡(jiǎn)直是完美,不過(guò)過(guò)了一段時(shí)間之后涌庭,發(fā)現(xiàn)好像還是有些瑕疵,使用構(gòu)造函數(shù)constructor模式的時(shí)候欧宜,函數(shù)里面的每個(gè)方法都會(huì)在每個(gè)實(shí)例上重新創(chuàng)建一遍,那么最明顯的地方是:

console.log(food1.sayName == food2.sayName); // 返回 false

因?yàn)槭褂?code>new來(lái)創(chuàng)建實(shí)例席镀,new的話(huà)還會(huì)把構(gòu)造函數(shù)里面的方法也一起創(chuàng)建,因?yàn)榉椒ㄒ彩呛瘮?shù)夏漱,而函數(shù)的實(shí)例化也會(huì)被new觸發(fā):

// 省略了其他部分,只關(guān)注方法部分
this.sayName = function() {
    console.log("我是" + this.name);
};

this.sayName = new function() {
    console.log("我是" + this.name);
}();

這樣就會(huì)造成內(nèi)存和時(shí)間和性能的浪費(fèi)挂绰,明明不需要重新重建新的函數(shù)實(shí)例的。

其實(shí)在之前的工廠模式里面葵蒂,也存在這個(gè)問(wèn)題,不過(guò)工廠模式更徹底践付,直接完全創(chuàng)建一個(gè)新對(duì)象,而構(gòu)造函數(shù)模式的話(huà)只是方法會(huì)被重新創(chuàng)建荔仁。

那怎么解決呢芽死?會(huì)用到原型模式,下回分解关贵。

參考內(nèi)容

  1. 紅寶書(shū),《javascript 高級(jí)程序設(shè)計(jì)第三版》

版權(quán)信息

作者: 慫如鼠
網(wǎng)站: https://www.whynotbetter.com
本作品著作權(quán)歸作者所有揖曾,商業(yè)轉(zhuǎn)載請(qǐng)聯(lián)系作者獲得授權(quán),非商業(yè)轉(zhuǎn)載請(qǐng)注明出處。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末炭剪,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子奴拦,更是在濱河造成了極大的恐慌,老刑警劉巖届吁,帶你破解...
    沈念sama閱讀 216,372評(píng)論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異暂氯,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)亮蛔,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)辣吃,“玉大人,你說(shuō)我怎么就攤上這事齿尽。” “怎么了灯节?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,415評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵绵估,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我国裳,道長(zhǎng),這世上最難降的妖魔是什么缝左? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,157評(píng)論 1 292
  • 正文 為了忘掉前任浓若,我火速辦了婚禮,結(jié)果婚禮上挪钓,老公的妹妹穿的比我還像新娘。我一直安慰自己耳舅,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,171評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布馏予。 她就那樣靜靜地躺著,像睡著了一般盔性。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上冕香,一...
    開(kāi)封第一講書(shū)人閱讀 51,125評(píng)論 1 297
  • 那天,我揣著相機(jī)與錄音暂筝,去河邊找鬼。 笑死焕襟,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的鸵赖。 我是一名探鬼主播,決...
    沈念sama閱讀 40,028評(píng)論 3 417
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼它褪,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了茫打?” 一聲冷哼從身側(cè)響起居触,我...
    開(kāi)封第一講書(shū)人閱讀 38,887評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤老赤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后抬旺,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,310評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡开财,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,533評(píng)論 2 332
  • 正文 我和宋清朗相戀三年误褪,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片兽间。...
    茶點(diǎn)故事閱讀 39,690評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡斋扰,死狀恐怖渡八,靈堂內(nèi)的尸體忽然破棺而出传货,到底是詐尸還是另有隱情屎鳍,我是刑警寧澤问裕,帶...
    沈念sama閱讀 35,411評(píng)論 5 343
  • 正文 年R本政府宣布逮壁,位于F島的核電站粮宛,受9級(jí)特大地震影響窥淆,放射性物質(zhì)發(fā)生泄漏巍杈。R本人自食惡果不足惜忧饭,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,004評(píng)論 3 325
  • 文/蒙蒙 一筷畦、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧鳖宾,春花似錦吼砂、人聲如沸鼎文。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,659評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至蚤假,卻和暖如春栏饮,著一層夾襖步出監(jiān)牢的瞬間磷仰,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,812評(píng)論 1 268
  • 我被黑心中介騙來(lái)泰國(guó)打工灶平, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留伺通,地道東北人逢享。 一個(gè)月前我還...
    沈念sama閱讀 47,693評(píng)論 2 368
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像瞒爬,于是被迫代替她去往敵國(guó)和親弓柱。 傳聞我的和親對(duì)象是個(gè)殘疾皇子侧但,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,577評(píng)論 2 353

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