Object.defineProperty()?完整指南示例詳解
Object.defineProperty() 完整指南
1. 基本概念
Object.defineProperty() 方法允許精確地添加或修改對象的屬性。默認情況下,使用此方法添加的屬性是不可修改的。
1.1 基本語法
Object.defineProperty(obj, prop, descriptor)
參數(shù)說明:
- obj: 要定義屬性的對象
- prop: 要定義或修改的屬性名
- descriptor: 屬性描述符對象
2. 屬性描述符
2.1 數(shù)據(jù)描述符
const obj = {}; Object.defineProperty(obj, 'name', { value: 'John', // 屬性值 writable: true, // 是否可寫 enumerable: true, // 是否可枚舉 configurable: true // 是否可配置 });
2.2 訪問器描述符
const obj = { _name: 'John' }; Object.defineProperty(obj, 'name', { get() { return this._name; }, set(value) { this._name = value; }, enumerable: true, configurable: true });
3. 實際應用示例
3.1 數(shù)據(jù)劫持(Vue2響應式原理)
function observe(obj) { if (typeof obj !== 'object' || obj === null) { return; } Object.keys(obj).forEach(key => { defineReactive(obj, key, obj[key]); }); } function defineReactive(obj, key, val) { // 遞歸處理嵌套對象 observe(val); Object.defineProperty(obj, key, { get() { console.log(`獲取${key}屬性`); return val; }, set(newVal) { if (val === newVal) return; console.log(`設置${key}屬性為${newVal}`); val = newVal; // 觸發(fā)更新 } }); } // 使用示例 const data = { name: 'John', age: 20 }; observe(data); data.name = 'Mike'; // 設置name屬性為Mike console.log(data.name); // 獲取name屬性 Mike
3.2 私有屬性模擬
function Person(name) { let _name = name; Object.defineProperty(this, 'name', { get() { return _name; }, set(value) { if (typeof value !== 'string') { throw new Error('Name must be a string'); } _name = value; } }); } const person = new Person('John'); console.log(person.name); // John person.name = 'Mike'; // 正常設置 person.name = 123; // 拋出錯誤
3.3 計算屬性實現(xiàn)
function computed(obj, key, computeFunc) { let value = computeFunc(); Object.defineProperty(obj, key, { get() { return value; }, set() { console.warn(`${key} is a computed property, cannot be modified`); } }); } const obj = { a: 1, b: 2 }; computed(obj, 'sum', () => obj.a + obj.b); console.log(obj.sum); // 3 obj.sum = 10; // 警告:sum is a computed property, cannot be modified
4. 注意事項和限制
4.1 不可擴展對象
const obj = {}; Object.preventExtensions(obj); // 這將拋出錯誤 Object.defineProperty(obj, 'name', { value: 'John' });
4.2 繼承屬性
const parent = {}; Object.defineProperty(parent, 'name', { value: 'John', writable: false }); const child = Object.create(parent); // 這將拋出錯誤 child.name = 'Mike';
4.3 屬性描述符限制
const obj = {}; // 不能同時指定 value/writable 和 get/set Object.defineProperty(obj, 'name', { value: 'John', get() { return 'John'; } }); // 拋出錯誤
5. 性能考慮
5.1 大量屬性處理
// 不推薦 const obj = {}; for (let i = 0; i < 1000; i++) { Object.defineProperty(obj, `prop${i}`, { value: i, writable: true }); } // 推薦 const descriptors = {}; for (let i = 0; i < 1000; i++) { descriptors[`prop${i}`] = { value: i, writable: true, configurable: true, enumerable: true }; } Object.defineProperties(obj, descriptors);
5.2 訪問器性能
// 避免在訪問器中進行復雜計算 Object.defineProperty(obj, 'name', { get() { // 不推薦 return complexCalculation(); } }); // 推薦:緩存計算結果 let cachedValue; Object.defineProperty(obj, 'name', { get() { if (cachedValue === undefined) { cachedValue = complexCalculation(); } return cachedValue; } });
6. 最佳實踐
描述符默認值
// 記住默認值都是 false Object.defineProperty(obj, 'name', { value: 'John' // writable: false // enumerable: false // configurable: false });
使用 TypeScript 類型
interface PropertyDescriptor { configurable?: boolean; enumerable?: boolean; value?: any; writable?: boolean; get?(): any; set?(v: any): void; }
錯誤處理
function safeDefineProperty(obj, prop, descriptor) { try { Object.defineProperty(obj, prop, descriptor); return true; } catch (error) { console.error(`Failed to define property ${prop}:`, error); return false; } }
7. 總結
Object.defineProperty() 的關鍵點:
使用場景
- 數(shù)據(jù)劫持
- 私有屬性模擬
- 計算屬性實現(xiàn)
- 屬性訪問控制
注意事項
- 描述符類型限制
- 性能考慮
- 繼承關系處理
- 錯誤處理
最佳實踐
- 合理使用緩存
- 避免復雜計算
- 注意默認值
- 做好錯誤處理
8. 深入理解 Object.defineProperty()
8.1 基礎概念詳解
Object.defineProperty() 是 JavaScript 中用于在對象上定義新屬性或修改現(xiàn)有屬性的方法。它允許精確控制屬性的特性。
// 基本語法 Object.defineProperty(obj, prop, descriptor) // 參數(shù)說明 // obj: 要定義屬性的對象 // prop: 要定義或修改的屬性名 // descriptor: 屬性描述符對象
8.2 屬性描述符詳解
屬性描述符分為兩種類型:數(shù)據(jù)描述符和訪問器描述符。
數(shù)據(jù)描述符的完整選項:
const obj = {}; Object.defineProperty(obj, 'name', { value: 'John', // 屬性值 writable: true, // 是否可寫 enumerable: true, // 是否可枚舉 configurable: true // 是否可配置 });
訪問器描述符的完整選項:
const obj = { _name: 'John' }; Object.defineProperty(obj, 'name', { get() { console.log('Getting value'); return this._name; }, set(value) { console.log('Setting value to', value); this._name = value; }, enumerable: true, configurable: true });
8.3 常見使用場景
只讀屬性:
const obj = {}; Object.defineProperty(obj, 'readonly', { value: 'I am read-only', writable: false, enumerable: true, configurable: false }); obj.readonly = 'New value'; // 無效 console.log(obj.readonly); // 'I am read-only'
不可枚舉屬性:
const obj = {}; Object.defineProperty(obj, 'hidden', { value: 'You cannot see me', enumerable: false }); console.log(Object.keys(obj)); // [] console.log(obj.hidden); // 'You cannot see me'
計算屬性:
const person = { firstName: 'John', lastName: 'Doe' }; Object.defineProperty(person, 'fullName', { get() { return `${this.firstName} ${this.lastName}`; }, set(value) { [this.firstName, this.lastName] = value.split(' '); } }); console.log(person.fullName); // 'John Doe' person.fullName = 'Jane Smith'; console.log(person.firstName); // 'Jane' console.log(person.lastName); // 'Smith'
Vue 雙向綁定實現(xiàn):
function observe(obj) { if (!obj || typeof obj !== 'object') return; // 遍歷對象的每個屬性 Object.keys(obj).forEach(key => { let value = obj[key]; let dep = new Dep(); // 依賴收集器 Object.defineProperty(obj, key, { get() { // 收集依賴 if (Dep.target) { dep.addDep(Dep.target); } return value; }, set(newValue) { if (value === newValue) return; value = newValue; // 通知所有依賴進行更新 dep.notify(); } }); // 遞歸觀察子屬性 if (typeof value === 'object') { observe(value); } }); } // 使用示例 const data = { user: { name: 'John', age: 20 } }; observe(data); // 現(xiàn)在 data 對象的所有屬性都是響應式的
8.4 注意事項和最佳實踐
描述符限制:
// 不能同時使用數(shù)據(jù)描述符和訪問器描述符 Object.defineProperty(obj, 'prop', { value: 123, get() { return 123; } // 錯誤! });
性能優(yōu)化:
// 批量定義屬性 Object.defineProperties(obj, { prop1: { value: 123, writable: true }, prop2: { get() { return this.prop1 * 2; } } });
默認值處理:
// 所有描述符屬性默認為 false Object.defineProperty(obj, 'prop', { value: 123 // writable: false // enumerable: false // configurable: false });
到此這篇關于Object.defineProperty() 完整指南的文章就介紹到這了,更多相關Object.defineProperty() 完整指南內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
javaScript如何跳出多重循環(huán)break、continue
這篇文章主要為大家詳細介紹了javaScript如何跳出多重循環(huán)break、continue,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-09-09JavaScript計算出現(xiàn)精度丟失問題的解決方法
Javascript作為一門大型編程語言,在日常開發(fā)中難免會涉及到大量的數(shù)學計算,然而,浮點數(shù)在計算過程中可能出現(xiàn)精度的問題,下面我們就來學習一下Javascript中高精度計算及其相關知識吧2023-11-11js創(chuàng)建數(shù)據(jù)共享接口——簡化框架之間相互傳值
很多框架存在父子關系,操作起來十分麻煩,很多同學經(jīng)常出現(xiàn)這樣悲催的代碼2011-10-10用headjs來管理和加載js 提高網(wǎng)站加載速度
headjs其實是一整套的工具,本文介紹的是它其中的Javascript Loader功能。需要的朋友可以參考下2016-11-11javascript實現(xiàn)五星評價代碼(源碼下載)
大家在淘寶購物之后,都會對賣家的服務進行評論,那么五星評價代碼是怎么實現(xiàn)的呢?下面小編給大家介紹基于Javascript實現(xiàn)五星評價代碼,有需要的朋友可以參考下2015-08-08