ECMAScript 5中的屬性描述符詳解_基礎知識
屬性描述符是ES5新增的概念,其作用是為物件的屬性增加更多的控制。
Object.defineProperty
要研究屬性描述符,首先要談談 Object.defineProperty 方法。這個方法的作用是給物件定義新屬性或修改已存在的屬性。其原型如下:
Object.defineProperty(obj, prop, descriptor)
使用範例:
var obj = { };
Object.defineProperty(obj, 'attr', { value: 1 });
上面一段程式碼為obj物件增加了一個名為attr的屬性,值為1。相當於:
var obj = { };
obj.attr = 1;
相較起來,Object.defineProperty 的寫法看似更複雜。但是,它最大的奧秘在於其第三個參數。
資料描述符
假設我們希望attr是一個唯讀屬性,就可以加上 writable 資料描述子:
var obj = { };
Object.defineProperty(obj, 'attr', {
value: 1,
writable: false
});
console.log(obj.attr);
obj.attr = 2; // fail
console.log(obj.attr);
執行以上程序可以發現,兩次印出來的attr的值都是1,也就是說對屬性的寫入失敗。然而,這樣的結果會有點莫名其妙,因為賦值語句的執行沒有異常,卻失敗了,試想如果在大片的程式碼中出現這樣的問題,就很難排查出來。事實上,只要以嚴格模式運行程式碼,就會產生異常:
'use strict'; // 進入嚴格模式
var obj = { };
Object.defineProperty(obj, 'attr', {
value: 1,
writable: false
});
obj.attr = 2; // throw exception
下面再來看看另一個資料描述符 enumerable ,它可以控制屬性是否能被列舉。如果只是簡單地定義一個屬性,這個屬性是可以在for...in迴圈中被列舉出來的:
var obj = { };
obj.attr = 1;
for (var i in obj) { console.log(obj[i]); }
enumerable 可以「藏」起來:
var obj = { };
Object.defineProperty(obj, 'attr', {
value: 1,
enumerable: false
});
for (var i in obj) { console.log(obj[i]); }
執行上面一段程式碼,會發現控制台什麼也沒輸出,因為此時attr屬性無法被列舉了。
講到這裡,大家可能有一個疑問,屬性描述符能否被修改?比方說一個唯讀屬性是否可以再定義為可寫?其實這取決於另一個資料描述符 configurable ,它可以控制屬性描述符能否被更改。
var obj = { };
Object.defineProperty(obj, 'attr', {
value: 1,
writable: false,
configurable: true
});
Object.defineProperty(obj, 'attr', {
writable: true
});
obj.attr = 2;
上面一段程式碼先把attr定義為唯讀屬性,然後再重新定義為可寫。所以對attr的寫入是成功的。
存取描述子
存取描述符類似物件導向中的get/set存取器。
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
在上面一段程式碼中,對attr的訪問事實上變成了對_attr的訪問,而且在set函數中限制了最小值為0。
取得屬性描述符
前面所述都是設定屬性描述符,那要如何取得已設定的描述符呢? Object.getOwnPropertyDescriptor 可以完成此項目工作。
var obj = { };
Object.defineProperty(obj, 'attr', {
value: 1,
writable: false,
configurable: true
});
var desc = Object.getOwnPropertyDescriptor(obj, 'attr');
console.dir(desc);
物件控制
前面說的 Object.defineProperty ,其操作的是物件的屬性,而下面說的三個方法則直接操作物件。
Object.preventExtensions 可以使物件無法擁有新的屬性:
var obj = { };
obj.attr = 1;
Object.preventExtensions(obj);
obj.attr2 = 2; //fail
Object.seal 可以讓物件僅剩下屬性值可以修改(如果屬性為唯讀,則連屬性值都無法修改):
var obj = { };
obj.attr = 1;
Object.seal(obj);
obj.attr = 1.5;
delete obj.attr; // fail
Object.freeze 可以使物件完全無法被修改:
var obj = { };
obj.attr = 1;
Object.freeze(obj);
obj.attr = 1.5; // fail
obj.attr2 = 2; //fail
然後大家可能又會問,怎麼知道某個物件是否曾經被preventExtensions、seal或freeze呢?答案就是分別呼叫 Object.isExtensible 、 Object.isSealed 、 Object.isFrozen ,這三個函數的用法比較簡單,就不再累贅了。
總的來說,透過屬性描述符可以進一步嚴格控制對象,加強程序邏輯的嚴謹性,唯一不足的就是,ES5在IE9裡面才基本實現(IE9還不支持嚴格模式),考慮到國內IE8份額還比較高的情況,這套東西目前只能在行動裝置瀏覽器和Node.js裡面用了。

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

熱門話題

本文討論了在瀏覽器中優化JavaScript性能的策略,重點是減少執行時間並最大程度地減少對頁面負載速度的影響。

本文討論了使用瀏覽器開發人員工具的有效JavaScript調試,專注於設置斷點,使用控制台和分析性能。

本文說明瞭如何使用源地圖通過將其映射回原始代碼來調試JAVASCRIPT。它討論了啟用源地圖,設置斷點以及使用Chrome DevTools和WebPack之類的工具。

本文探討了Java收藏框架的有效使用。 它強調根據數據結構,性能需求和線程安全選擇適當的收集(列表,設置,地圖,隊列)。 通過高效優化收集用法

掌握了入門級TypeScript教程後,您應該能夠在支持TypeScript的IDE中編寫自己的代碼,並將其編譯成JavaScript。本教程將深入探討TypeScript中各種數據類型。 JavaScript擁有七種數據類型:Null、Undefined、Boolean、Number、String、Symbol(ES6引入)和Object。 TypeScript在此基礎上定義了更多類型,本教程將詳細介紹所有這些類型。 Null數據類型 與JavaScript一樣,TypeScript中的null

本教程將介紹如何使用 Chart.js 創建餅圖、環形圖和氣泡圖。此前,我們已學習了 Chart.js 的四種圖表類型:折線圖和條形圖(教程二),以及雷達圖和極地區域圖(教程三)。 創建餅圖和環形圖 餅圖和環形圖非常適合展示某個整體被劃分為不同部分的比例。例如,可以使用餅圖展示野生動物園中雄獅、雌獅和幼獅的百分比,或不同候選人在選舉中獲得的投票百分比。 餅圖僅適用於比較單個參數或數據集。需要注意的是,餅圖無法繪製值為零的實體,因為餅圖中扇形的角度取決於數據點的數值大小。這意味著任何占比為零的實體
