JavaScript創(chuàng)建對象的方式

1.使用工廠模式創(chuàng)建

使用特定的函數(shù)創(chuàng)建對象。

function createClass(name, age) {
    let o = new Object();
    o.name = name;
    o.age = age;
    o.say = function() {
        console.log(this.name, this.age)
    }
    return o;
}

let c = createClass('xxx', 11);
c.say();

函數(shù)createClass接收兩個參數(shù)虽填,然后函數(shù)的內(nèi)部通過這兩個參數(shù)創(chuàng)建出一個對象返回丁恭。
這種創(chuàng)建方式,有個問題斋日,就是不知道內(nèi)部創(chuàng)建的對象是什么類型的牲览。

2.構(gòu)造函數(shù)創(chuàng)建

使用構(gòu)造函數(shù)模式,可以這樣創(chuàng)建對象恶守。

function CustomClass(name, age) {
    this.name = name;
    this.age = age;
    this.say = function() {
        console.log(this.name, this.age);
    }
}
let c = new CustomClass('xxx', 11);
c.say();

通過CustomClass()構(gòu)造函數(shù)代替了createClass()工廠函數(shù)第献。實際上內(nèi)部是一致的,區(qū)別在于:

  • 1.沒有顯示的創(chuàng)建對象兔港。
  • 2.屬性和方法直接賦值給了this庸毫。
  • 3.沒有return。
  • 4.函數(shù)名首字母大寫了衫樊,這是一種約定俗成的寫法飒赃,表示是一個對象。

new關(guān)鍵字的作用:

  • 1.創(chuàng)建空對象{}
  • 2.this指向這個空對象 this={}
  • 3.給對象賦值
  • 4.返回這個對象 return this

通過這種方式創(chuàng)建的對象科侈,會有一個屬性constructor指向了CustomClass载佳,如下所示:

console.log(c.constructor);
console.log(c.constructor === CustomClass);
輸出:
? CustomClass(name, age) {
    this.name = name;
    this.age = age;
    this.say = function() {
        console.log(this.name, this.age);
    }
}
true

這樣,我們就可以通過constructor知道對象的類型兑徘。不過一般來說刚盈,都是通過instanceof操作符來判斷對象的類型的羡洛。如下:

console.log(c instanceof CustomClass)
console.log(c instanceof Object)
輸出:
true
true

可以看到挂脑,都是true,之所以O(shè)bject也是true欲侮,那是因為所有的對象都繼承自O(shè)bject崭闲。

構(gòu)造函數(shù)可以賦值給變量,然后通過變量來創(chuàng)建對象威蕉。

const CustomClass = function(name, age) {
    this.name = name;
    this.age = age;
    this.say = function() {
        console.log(this.name, this.age);
    }
}
let c = new CustomClass('xxx', 11);
c.say();

調(diào)用構(gòu)造函數(shù)的時候刁俭,參數(shù)可以不寫,只要有new操作符韧涨,就可以調(diào)用構(gòu)造函數(shù)創(chuàng)建對象牍戚。

let c = new CustomClass();

構(gòu)造函數(shù)和普通函數(shù)沒有區(qū)別,唯一的不同就是調(diào)用時候的不同虑粥。構(gòu)造函數(shù)前面如果添加了new操作符如孝,那么他就會創(chuàng)建出一個對象,否則娩贷,他就是一個普通的函數(shù)第晰。


通過構(gòu)造函數(shù)創(chuàng)建的對象,會有一個問題茁瘦,就是對象的方法每次創(chuàng)建對象時,都是新建一個方法圆恤,然后賦值給對象哑了。這樣就會導(dǎo)致不同的對象弱左,相同的方法不是同一個對象炕淮。如下所示:

let c1 = new CustomClass();
let c2 = new CustomClass();
console.log(c1.say === c2.say)
輸出:
false

這是可以解決的涂圆,我們只需要讓函數(shù)只創(chuàng)建一次就可以了润歉,如下所示:

function CustomClass(name, age) {
    this.name = name;
    this.age = age;
    this.say = say
}

function say() {
    console.log(this.name, this.age);
}

在外部創(chuàng)建,然后內(nèi)部引用同一個函數(shù)嚼鹉,可以解決上述問題驱富。但是這樣非常容易造成需要在外部定義多個函數(shù)褐鸥,導(dǎo)致代碼可能會被分散在各個地方叫榕。

3.原型模式創(chuàng)建

每一個函數(shù)晰绎,都有一個prototype屬性寒匙,我們叫做原型躏将。我們可以通過prototype屬性祸憋,將需要在構(gòu)造函數(shù)內(nèi)賦值的值蚯窥,在外部賦值給prototype屬性拦赠,這樣荷鼠,創(chuàng)建出來的對象實例就都會通過原型獲得這些值允乐。

function Person() {}
Person.prototype.name = "哈哈哈";
Person.prototype.age = 29;
Person.prototype.sayName = function() {
    console.log(this.name);
};
let person1 = new Person();
person1.sayName(); // "哈哈哈"
let person2 = new Person();
person2.sayName(); // "哈哈哈"
console.log(person1.sayName == person2.sayName); // true

我們需要了解到牍疏,無論何時拨齐,只要創(chuàng)建出一個函數(shù)瞻惋,就會給該函數(shù)創(chuàng)建出一個prototype屬性熟史,指向了原型對象窄俏。默認情況下限寞,所有的原型對象會自動獲得一個constructor屬性仰坦,指向與之關(guān)聯(lián)的構(gòu)造函數(shù)悄晃。
然后在調(diào)用屬性或者方法的時候玫霎,會先在對象實例本身進行搜索凿滤,如果實例上有該方法屬性,則執(zhí)行庶近,沒有則會沿著prototype找到原型對象翁脆,如果原型對象上有,則執(zhí)行鼻种。


in操作符可以判斷屬性是否可用反番。該屬性在實例或者原型上,都會返回true叉钥。
hasOwnProperty方法罢缸,如果屬性在原型上,則會返回true投队。

4.通過類創(chuàng)建

使用關(guān)鍵字class聲明類枫疆,然后創(chuàng)建出對象。

// 類聲明
class Person {}
// 類表達式
const Animal = class {}

可使用constructor關(guān)鍵字敷鸦,在類定義內(nèi)部創(chuàng)建類的構(gòu)造函數(shù)养铸,constructor會告訴解釋器,在使用new操作符創(chuàng)建新實例時轧膘,調(diào)用該函數(shù)钞螟。如下:

class Person {
    constructor() {
        console.log('init')
    }
}
let p = new Person() // init

可在實例創(chuàng)建時,傳入?yún)?shù)谎碍,也可以不傳鳞滨,不傳參數(shù)時,創(chuàng)建的時候類后面的()也可以省略蟆淀。如下:

class Person {
    constructor(name) {
        this.name = name
        console.log('init', name)
    }
}
let p = new Person() // init undefined
let p2 = new Person("aa") // init aa
let p3 = new Person // init undefined

類構(gòu)造和構(gòu)造函數(shù)構(gòu)造主要區(qū)別是拯啦,類構(gòu)造必須使用new操作符,構(gòu)造函數(shù)的話熔任,不寫new操作符褒链,那就是一個普通的函數(shù),類構(gòu)造不寫則會報錯疑苔。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末甫匹,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子惦费,更是在濱河造成了極大的恐慌兵迅,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,968評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件薪贫,死亡現(xiàn)場離奇詭異恍箭,居然都是意外死亡,警方通過查閱死者的電腦和手機瞧省,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評論 2 382
  • 文/潘曉璐 我一進店門扯夭,熙熙樓的掌柜王于貴愁眉苦臉地迎上來鳍贾,“玉大人,你說我怎么就攤上這事交洗〖致” “怎么了?”我有些...
    開封第一講書人閱讀 153,220評論 0 344
  • 文/不壞的土叔 我叫張陵藕筋,是天一觀的道長纵散。 經(jīng)常有香客問我,道長隐圾,這世上最難降的妖魔是什么伍掀? 我笑而不...
    開封第一講書人閱讀 55,416評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮暇藏,結(jié)果婚禮上蜜笤,老公的妹妹穿的比我還像新娘。我一直安慰自己盐碱,他們只是感情好把兔,可當我...
    茶點故事閱讀 64,425評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著瓮顽,像睡著了一般县好。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上暖混,一...
    開封第一講書人閱讀 49,144評論 1 285
  • 那天缕贡,我揣著相機與錄音,去河邊找鬼拣播。 笑死晾咪,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的贮配。 我是一名探鬼主播谍倦,決...
    沈念sama閱讀 38,432評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼泪勒!你這毒婦竟也來了昼蛀?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,088評論 0 261
  • 序言:老撾萬榮一對情侶失蹤酣藻,失蹤者是張志新(化名)和其女友劉穎曹洽,沒想到半個月后鳍置,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體辽剧,經(jīng)...
    沈念sama閱讀 43,586評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,028評論 2 325
  • 正文 我和宋清朗相戀三年税产,在試婚紗的時候發(fā)現(xiàn)自己被綠了怕轿。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片偷崩。...
    茶點故事閱讀 38,137評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖撞羽,靈堂內(nèi)的尸體忽然破棺而出阐斜,到底是詐尸還是另有隱情,我是刑警寧澤诀紊,帶...
    沈念sama閱讀 33,783評論 4 324
  • 正文 年R本政府宣布谒出,位于F島的核電站,受9級特大地震影響邻奠,放射性物質(zhì)發(fā)生泄漏笤喳。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,343評論 3 307
  • 文/蒙蒙 一碌宴、第九天 我趴在偏房一處隱蔽的房頂上張望杀狡。 院中可真熱鬧,春花似錦贰镣、人聲如沸呜象。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽恭陡。三九已至,卻和暖如春上煤,著一層夾襖步出監(jiān)牢的瞬間子姜,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評論 1 262
  • 我被黑心中介騙來泰國打工楼入, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留哥捕,地道東北人。 一個月前我還...
    沈念sama閱讀 45,595評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親牺勾。 傳聞我的和親對象是個殘疾皇子涨薪,可洞房花燭夜當晚...
    茶點故事閱讀 42,901評論 2 345

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