js中創(chuàng)建對(duì)象的幾種方式
前言
不管是哪門(mén)語(yǔ)言,千變?nèi)f化不離其宗,深入理解其本質(zhì),方能應(yīng)用自如。對(duì)應(yīng)到j(luò)s,閉包,原型,函數(shù),對(duì)象等是需要花費(fèi)大功夫思考、理解的。本文穿插了js原型和函數(shù)的相關(guān)知識(shí),討論了批量創(chuàng)建對(duì)象的幾種方式以及它們的優(yōu)缺點(diǎn)。
正文
說(shuō)起創(chuàng)建對(duì)象,最容易想到的便是通過(guò)對(duì)象字面量方式直接定義一個(gè)對(duì)象吧,但這種方式只能創(chuàng)建少量,單獨(dú)且相互間無(wú)聯(lián)系的對(duì)象。若要批量創(chuàng)建對(duì)象,該如何?
工廠(chǎng)模式
工廠(chǎng)模式非常直觀(guān),將創(chuàng)建對(duì)象的過(guò)程抽象為一個(gè)函數(shù),用函數(shù)封裝以特定接口創(chuàng)建對(duì)象的細(xì)節(jié)。如下所示:
function createStudent(name,sex,grade){ var o = new Object(); o.name = name; o.sex = sex; o.grade = grade; o.sayName = function(){ console.log(this.name); } return o; } var s1 = createStudent('Claiyre','famale',1);
通俗地講,工廠(chǎng)模式就是將創(chuàng)建對(duì)象的語(yǔ)句放在一個(gè)函數(shù)里,通過(guò)傳入?yún)?shù)來(lái)創(chuàng)建特定對(duì)象,最后返回創(chuàng)建的對(duì)象。
工廠(chǎng)模式雖然可以創(chuàng)建多個(gè)相似的對(duì)象,但卻不能解決對(duì)象標(biāo)識(shí)的問(wèn)題,即怎樣知道一個(gè)對(duì)象的類(lèi)型。構(gòu)造函數(shù)模式應(yīng)運(yùn)而生。
構(gòu)造函數(shù)模式
構(gòu)造函數(shù)模式是java語(yǔ)言創(chuàng)建對(duì)象的通用方式。兩種語(yǔ)言用構(gòu)造函數(shù)創(chuàng)建對(duì)象的方式略有不同,注意區(qū)別。
在JavaScript中沒(méi)有類(lèi)的概念,函數(shù)即為一等公民,因此,不必顯式聲明某個(gè)類(lèi),直接創(chuàng)建構(gòu)造函數(shù)即可,類(lèi)的方法和屬性在構(gòu)造函數(shù)中(或原型對(duì)象上)處理。構(gòu)造函數(shù)模式的示例代碼如下:
function Student(name,sex,grade){ this.name = name; this.sex = sex; this.grade = grade; this.sayName = function(){ console.log(this.name); } } var s2 = new Student('孫悟空','male',2);
細(xì)心的朋友一定發(fā)現(xiàn)了構(gòu)造函數(shù)的函數(shù)名首字母是大寫(xiě)的,而普通函數(shù)首字母則是小寫(xiě),這是眾多OO語(yǔ)言約定俗成的規(guī)定,雖然大多數(shù)情況下不大寫(xiě)也不會(huì)報(bào)錯(cuò),但是為了代碼的規(guī)范性和可讀性,還是應(yīng)該將構(gòu)造函數(shù)的首字母大寫(xiě),與普通函數(shù)區(qū)別開(kāi)。
與工廠(chǎng)模式相比,用構(gòu)造模式創(chuàng)建對(duì)象有以下幾點(diǎn)不同:
- 沒(méi)有顯示地創(chuàng)建對(duì)象
- 直接將屬性和方法賦給this對(duì)象
- 沒(méi)有return語(yǔ)句
此外,還應(yīng)注意到要?jiǎng)?chuàng)建Student的實(shí)例,必須要使用new操作符,創(chuàng)建的實(shí)例對(duì)象將有一個(gè)constructor(構(gòu)造器)屬性,指向Person構(gòu)造函數(shù)。調(diào)用構(gòu)造函數(shù)創(chuàng)建對(duì)象經(jīng)過(guò)了以下幾個(gè)過(guò)程:
- 創(chuàng)建一個(gè)新對(duì)象
- 將構(gòu)造函數(shù)的作用域賦給新對(duì)象(因此this就指向了這個(gè)新對(duì)象)
- 執(zhí)行構(gòu)造函數(shù)中的代碼
- 返回新對(duì)象(不需要顯式返回)
構(gòu)造函數(shù)雖好用,但也不是沒(méi)有缺點(diǎn)。使用構(gòu)造函數(shù)的主要問(wèn)題是:每個(gè)方法都要在每個(gè)實(shí)例上創(chuàng)建一遍。在ECMAScript中,函數(shù)即對(duì)象,因此每定義一個(gè)函數(shù),也就是實(shí)例化了一個(gè)對(duì)象。下面的例子證明了這個(gè)缺點(diǎn)。
var s3 = new Student('唐僧','male',3); var s4 = new Student('白骨精','female',4); s3.sayName(); s4.sayName(); console.log(s3.sayName == s4.sayName);
運(yùn)行結(jié)果:
也就是說(shuō)通過(guò)構(gòu)造函數(shù)實(shí)例化的多個(gè)對(duì)象的方法,是多個(gè)不同的方法,但它們內(nèi)部的代碼以及實(shí)現(xiàn)的功能是相同的,這就造成了一定的資源浪費(fèi)。
幸運(yùn)的是,這個(gè)問(wèn)題可以用原型模式來(lái)解決。
原型模式
js中,每個(gè)函數(shù)都有一個(gè)prototype屬性,它是一個(gè)指針,指向一個(gè)對(duì)象,叫做原型對(duì)象,原型對(duì)象包含了可以由特定類(lèi)型的所有實(shí)例對(duì)象共享的屬性和方法。此外,這個(gè)對(duì)象有一個(gè)與生自來(lái)的屬性constructor,指向創(chuàng)建對(duì)象的構(gòu)造方法。
使用原型模式可以讓所有的實(shí)例共享原型對(duì)象中的屬性和方法,也就是說(shuō),不必再構(gòu)造函數(shù)中定義對(duì)象實(shí)例的信息。用代碼表示如下:
function Student_1(){ } Student_1.prototype.name = 'Claiyre'; Student_1.prototype.sex = 'female'; Student_1.prototype.class = 5; Student_1.prototype.sayName = function (){ console.log(this.name); } var s5 = new Student_1(); s5.sayName(); //Claiyre var s6 = new Student_1(); s6.sayName(); //Claiyre
一張圖勝過(guò)千言萬(wàn)語(yǔ),下圖清楚地闡釋了各個(gè)對(duì)象和原型對(duì)象間的關(guān)系:
了解過(guò)原型后,可以繼續(xù)在實(shí)例對(duì)象上增添屬性或方法:
s6.name = 'John'; s6.sayName(); //John
當(dāng)要讀取某個(gè)對(duì)象的屬性時(shí),都會(huì)執(zhí)行一次搜索,搜索首先從對(duì)象實(shí)例本身開(kāi)始,如果在實(shí)例中找到了這個(gè)屬性,則搜索結(jié)束,返回實(shí)例屬性的值;若實(shí)例上沒(méi)有找到,則繼續(xù)向?qū)ο蟮脑蛯?duì)象延伸,搜索對(duì)象的原型對(duì)象,若在原型對(duì)象上找到了,則返回原型上相應(yīng)屬性的值,若沒(méi)有找到,則返回undefined。因此,實(shí)例對(duì)象屬性會(huì)覆蓋原型對(duì)象上的同名屬性,所以上面第二行代碼輸出的是John。
- Object.getPrototypeOf(object)方法返回參數(shù)對(duì)象的原型對(duì)象。
- Object.keys(object)方法返回對(duì)象上課枚舉的實(shí)例屬性。
原型中的所有屬性都是被所有實(shí)例所共享的,這種共享對(duì)于函數(shù)來(lái)說(shuō)非常合適,對(duì)于包含基本值的屬性也說(shuō)的過(guò)去(實(shí)例屬性會(huì)覆蓋原型同名屬性),但對(duì)于那些包含引用類(lèi)型的屬性,可有大麻煩了
Student_1.prototype.friends = ['aa','bb']; console.log('s6的朋友' + s6.friends); s5.friends.push('cc'); console.log('s5的朋友' + s5.friends); console.log('s6的朋友' + s6.friends);
運(yùn)行結(jié)果:
問(wèn)題來(lái)了,我們只想改變s5的朋友列表,但由于原型模式的共享本質(zhì),s6的朋友列表也隨之改變了。
因此,很少單獨(dú)使用原型模式。
組合使用構(gòu)造函數(shù)和原型模式
構(gòu)造函數(shù)模式用于定義實(shí)例屬性,原型模式則用于定義方法和共享的屬性。這種混合模式不僅支持向構(gòu)造函數(shù)傳入?yún)?shù),還最大限度地節(jié)約了內(nèi)存,可謂是集兩模式之長(zhǎng)。示例代碼如下:
function Student(name,sex,grade){ this.name = name; this.sex = sex; this.grade = grade; } Student.prototype.sayName = function(){ console.log(this.name); } Student.prototype.school = 'Joooh school';
其他模式
除了以上幾種常見(jiàn)的模式外,批量創(chuàng)建對(duì)象的方式還有
- 動(dòng)態(tài)原型模式:僅在第一次調(diào)用構(gòu)造函數(shù)時(shí),將方法賦給原型對(duì)象的相應(yīng)屬性,其他示例的處理方式同構(gòu)造函數(shù)模式
- 寄生構(gòu)造函數(shù)模式:僅僅封裝創(chuàng)建對(duì)象的代碼,然后再返回新創(chuàng)建的對(duì)象,仍使用new操作符調(diào)用
- 穩(wěn)妥構(gòu)造函數(shù)模式:沒(méi)有公共屬性,只有私有變量和方法,以及一些get/set方法,用以處理私有變量。
結(jié)語(yǔ)
每種模式都有各自的優(yōu)缺點(diǎn),具體要使用哪種,還需結(jié)合實(shí)際場(chǎng)景,深入理解,靈活運(yùn)用。
以上就是本文的全部?jī)?nèi)容,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,同時(shí)也希望多多支持腳本之家!
- JS中的函數(shù)與對(duì)象的創(chuàng)建方式
- JavaScript創(chuàng)建對(duì)象方式總結(jié)【工廠(chǎng)模式、構(gòu)造函數(shù)模式、原型模式等】
- JavaScript創(chuàng)建對(duì)象的常用方式總結(jié)
- JavaScript創(chuàng)建對(duì)象的七種方式(推薦)
- 深入理解JavaScript創(chuàng)建對(duì)象的多種方式以及優(yōu)缺點(diǎn)
- JS對(duì)象創(chuàng)建的幾種方式整理
- 創(chuàng)建一般js對(duì)象的幾種方式
- 詳解js創(chuàng)建對(duì)象的幾種方式和對(duì)象方法
相關(guān)文章
js實(shí)現(xiàn)隨機(jī)點(diǎn)名系統(tǒng)(實(shí)例講解)
下面小編就為大家?guī)?lái)一篇js實(shí)現(xiàn)隨機(jī)點(diǎn)名系統(tǒng)(實(shí)例講解)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-10-10js 按照指定間隔 向字符串中插入隨機(jī)字符串的實(shí)現(xiàn)代碼
看到論壇有人問(wèn),覺(jué)得有意思,就試著寫(xiě)了一下。2010-03-03javascript實(shí)現(xiàn)簡(jiǎn)單的html5視頻播放器
網(wǎng)頁(yè)視頻音頻播放器大家并不陌生,在IE中我們可以運(yùn)行ActiveX來(lái)嵌入微軟的Media Player或者其他的本地播放器,當(dāng)然可能大部分我們都是使用Flash來(lái)制作播放器。在HTML5發(fā)展迅速的今天,讓我們嘗試用HTML5來(lái)制作網(wǎng)頁(yè)播放器吧,畢竟無(wú)論是PC還是移動(dòng)設(shè)備,HTML5是未來(lái)的趨勢(shì)2015-05-05改變文件域的樣式實(shí)現(xiàn)思路同時(shí)兼容ie、firefox
正如標(biāo)題所言只是模擬了file文件域的外觀(guān),其實(shí)起作用的還是文件域file,這樣就很方便,感興趣的朋友可以了解下2013-10-10Echarts基本用法_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
這篇文章主要介紹了Echarts基本用法,詳解的介紹了Echarts的基本用法和實(shí)例,有興趣的可以了解一下2017-08-08JS的location.href跳出框架打開(kāi)新頁(yè)面的方法
登錄頁(yè)面在框架內(nèi)打開(kāi),想讓它直接跳出框架打開(kāi)(這里不是打開(kāi)新窗口),終于在網(wǎng)上找到了辦法,下面分享給大家2014-09-09JavaScript中統(tǒng)計(jì)Textarea字?jǐn)?shù)并提示還能輸入的字符
是在文本框中輸入文字的時(shí)候,會(huì)自動(dòng)統(tǒng)計(jì)輸入的字符,并顯示用戶(hù)還能輸入的字符,其實(shí)js也可以實(shí)現(xiàn),下面就以示例的方式為大家講解下2014-06-06