ECMAScript 5中的屬性描述符詳解
來(lái)源:易賢網(wǎng) 閱讀:843 次 日期:2015-03-06 11:41:37
溫馨提示:易賢網(wǎng)小編為您整理了“ECMAScript 5中的屬性描述符詳解”,方便廣大網(wǎng)友查閱!

這篇文章主要介紹了ECMAScript 5中的屬性描述符詳解,本文講解了Object.defineProperty、數(shù)據(jù)描述符、存取描述符、獲取屬性描述符、對(duì)象控制等內(nèi)容,需要的朋友可以參考下

屬性描述符是ES5中新增的概念,其作用是給對(duì)象的屬性增加更多的控制。

Object.defineProperty

要研究屬性描述符,首先要談?wù)?Object.defineProperty 方法。這個(gè)方法的作用是給對(duì)象定義新屬性或修改已存在的屬性。其原型如下:

代碼如下:

Object.defineProperty(obj, prop, descriptor)

使用示例:

代碼如下:

var obj = { };

Object.defineProperty(obj, 'attr', { value: 1 });

上面一段代碼給obj對(duì)象增加了一個(gè)名為attr的屬性,值為1。相當(dāng)于:

代碼如下:

var obj = { };

obj.attr = 1;

相比起來(lái),Object.defineProperty 的寫(xiě)法看似更為復(fù)雜。但是,它最大的奧秘在于其第三個(gè)參數(shù)。

數(shù)據(jù)描述符

假設(shè)我們希望attr是一個(gè)只讀屬性,就可以加上 writable 數(shù)據(jù)描述符:

代碼如下:

var obj = { };

Object.defineProperty(obj, 'attr', {

value: 1,

writable: false

});

console.log(obj.attr);

obj.attr = 2; // fail

console.log(obj.attr);

執(zhí)行以上程序可以發(fā)現(xiàn),兩次打印出來(lái)的attr的值都是1,也就是說(shuō)對(duì)屬性的寫(xiě)入失敗。然而,這樣的結(jié)果會(huì)有點(diǎn)莫名其妙,因?yàn)橘x值語(yǔ)句的執(zhí)行沒(méi)有異常,卻失敗了,試想如果在大片的代碼中出現(xiàn)這樣的問(wèn)題,就很難排查出來(lái)。事實(shí)上,只要以嚴(yán)格模式運(yùn)行代碼,就會(huì)產(chǎn)生異常:

復(fù)制代碼 代碼如下:

'use strict'; // 進(jìn)入嚴(yán)格模式

var obj = { };

Object.defineProperty(obj, 'attr', {

value: 1,

writable: false

});

obj.attr = 2; // throw exception

下面再來(lái)看看另一個(gè)數(shù)據(jù)描述符 enumerable ,它可以控制屬性是否能被枚舉。如果只是簡(jiǎn)單地定義一個(gè)屬性,這個(gè)屬性是可以在for...in循環(huán)中被枚舉出來(lái)的:

代碼如下:

var obj = { };

obj.attr = 1;

for (var i in obj) { console.log(obj[i]); }

enumerable 可以將其“藏”起來(lái):

var obj = { };

Object.defineProperty(obj, 'attr', {

value: 1,

enumerable: false

});

for (var i in obj) { console.log(obj[i]); }

執(zhí)行上面一段代碼,會(huì)發(fā)現(xiàn)控制臺(tái)什么也沒(méi)輸出,因?yàn)榇藭r(shí)attr屬性無(wú)法被枚舉了。

講到這里,大家可能有一個(gè)疑問(wèn),屬性描述符能否被修改?比方說(shuō)一個(gè)只讀屬性是否可以再次定義為可寫(xiě)?其實(shí)這取決于另一個(gè)數(shù)據(jù)描述符 configurable ,它可以控制屬性描述符能否被更改。

代碼如下:

var obj = { };

Object.defineProperty(obj, 'attr', {

value: 1,

writable: false,

configurable: true

});

Object.defineProperty(obj, 'attr', {

writable: true

});

obj.attr = 2;

上面一段代碼先把a(bǔ)ttr定義為只讀屬性,然后又重新定義為可寫(xiě)。所以對(duì)attr的寫(xiě)入是成功的。

存取描述符

存取描述符類(lèi)似面向?qū)ο笾械膅et/set訪問(wèn)器。

代碼如下:

var obj = { };

Object.defineProperty(obj, 'attr', {

set: function(val) { this._attr = Math.max(0, val); },

get: function() { return this._attr; }

});

obj.attr = -1;

console.log(obj.attr); // 0

在上面一段代碼中,對(duì)attr的訪問(wèn)事實(shí)上變成了對(duì)_attr的訪問(wèn),而且在set函數(shù)中限制了最小值為0。

獲取屬性描述符

前面所述都是設(shè)置屬性描述符,那如何獲取已設(shè)置的描述符呢?Object.getOwnPropertyDescriptor 可以完成此項(xiàng)工作。

代碼如下:

var obj = { };

Object.defineProperty(obj, 'attr', {

value: 1,

writable: false,

configurable: true

});

var desc = Object.getOwnPropertyDescriptor(obj, 'attr');

console.dir(desc);

對(duì)象控制

前面說(shuō)的 Object.defineProperty ,其操作的是對(duì)象的屬性,而下面說(shuō)的三個(gè)方法則直接操作對(duì)象。

Object.preventExtensions 可以使對(duì)象無(wú)法擁有新的屬性:

代碼如下:

var obj = { };

obj.attr = 1;

Object.preventExtensions(obj);

obj.attr2 = 2; //fail

Object.seal 可以使對(duì)象僅剩屬性值可以修改(如果屬性為只讀,則連屬性值都無(wú)法修改):

代碼如下:

var obj = { };

obj.attr = 1;

Object.seal(obj);

obj.attr = 1.5;

delete obj.attr; // fail

Object.freeze 可以使對(duì)象完全無(wú)法被修改:

代碼如下:

var obj = { };

obj.attr = 1;

Object.freeze(obj);

obj.attr = 1.5; // fail

obj.attr2 = 2; //fail

然后大家可能又會(huì)問(wèn),怎么知道某個(gè)對(duì)象是否曾經(jīng)被preventExtensions、seal或者freeze呢?答案就是分別調(diào)用 Object.isExtensible 、 Object.isSealed 、 Object.isFrozen ,這三個(gè)函數(shù)的用法比較簡(jiǎn)單,就不再累贅了。

總的來(lái)說(shuō),通過(guò)屬性描述符可以進(jìn)一步嚴(yán)格控制對(duì)象,加強(qiáng)程序邏輯的嚴(yán)謹(jǐn)性,唯一不足的就是,ES5在IE9里面才基本實(shí)現(xiàn)(IE9還不支持嚴(yán)格模式),考慮到國(guó)內(nèi)IE8份額還比較高的情況,這套東西目前只能在移動(dòng)端瀏覽器和Node.js里面用了。

更多信息請(qǐng)查看IT技術(shù)專(zhuān)欄

更多信息請(qǐng)查看腳本欄目
易賢網(wǎng)手機(jī)網(wǎng)站地址:ECMAScript 5中的屬性描述符詳解
由于各方面情況的不斷調(diào)整與變化,易賢網(wǎng)提供的所有考試信息和咨詢(xún)回復(fù)僅供參考,敬請(qǐng)考生以權(quán)威部門(mén)公布的正式信息和咨詢(xún)?yōu)闇?zhǔn)!

2025國(guó)考·省考課程試聽(tīng)報(bào)名

  • 報(bào)班類(lèi)型
  • 姓名
  • 手機(jī)號(hào)
  • 驗(yàn)證碼
關(guān)于我們 | 聯(lián)系我們 | 人才招聘 | 網(wǎng)站聲明 | 網(wǎng)站幫助 | 非正式的簡(jiǎn)要咨詢(xún) | 簡(jiǎn)要咨詢(xún)須知 | 加入群交流 | 手機(jī)站點(diǎn) | 投訴建議
工業(yè)和信息化部備案號(hào):滇ICP備2023014141號(hào)-1 云南省教育廳備案號(hào):云教ICP備0901021 滇公網(wǎng)安備53010202001879號(hào) 人力資源服務(wù)許可證:(云)人服證字(2023)第0102001523號(hào)
云南網(wǎng)警備案專(zhuān)用圖標(biāo)
聯(lián)系電話:0871-65099533/13759567129 獲取招聘考試信息及咨詢(xún)關(guān)注公眾號(hào):hfpxwx
咨詢(xún)QQ:526150442(9:00—18:00)版權(quán)所有:易賢網(wǎng)
云南網(wǎng)警報(bào)警專(zhuān)用圖標(biāo)