Javascript中的對象屬性是有序的嗎
前言
最近有人問我,JavaScript對象屬性是否一定是無序的、不可預測的?
早期接觸過JavaScript的開發者可能會回答,Object.keys()
或for...in
會返回一個不可預知的對象屬性順序。
但現在的情況仍然是這樣嗎?
不是瞭,有些情況下是有序的。
從ECMAScript 2020開始,Object.key
、for...in
、Object.getOwnPropertyNames
和Reflect.ownKeys
都遵循同一個規范順序。它們是:
自己的屬性是數組的索引,按數字索引升序排列
const obj = { 100: 100, '2': 2, 12: 12, '0': 0 } // 下面打印的結果順序都是 ['0', '2', '12', '100'] console.log(Object.keys(obj)) console.log(Object.getOwnPropertyNames(obj)) console.log(Reflect.ownKeys(obj)) for (const key in obj) { console.log('key', key) }
const obj = { a: 'a', }; obj.b = 'b'; setTimeout(() => { obj.c = 'c'; }); obj.d = 'd'; // 下面打印的結果順序都是 `[ 'a', 'b', 'd' ]` console.log(Object.keys(obj)); console.log(Object.getOwnPropertyNames(obj)); console.log(Reflect.ownKeys(obj)); for (const key in obj) { console.log('key: ', key); }
上面的代碼添加瞭事件循環的知識點。因為 setTimeout
是一個異步的宏任務,當console.log
輸出時,c
屬性還沒有被添加到 obj
中。
自身的 Symbol 屬性,按創建時間順序遞增
const obj = { [Symbol('a')]: 'a', [Symbol.for('b')]: 'b', }; obj[Symbol('c')] = 'c'; console.log(Object.keys(obj)); // [] console.log(Object.getOwnPropertyNames(obj)); // [] console.log(Reflect.ownKeys(obj)); // [ Symbol(a), Symbol(b), Symbol(c) ] for (const key in obj) { console.log('key: ', key); // 沒有輸出 } console.log(Object.getOwnPropertySymbols(obj)); // [ Symbol(a), Symbol(b), Symbol(c) ]
Symbol 屬性和 String 屬性一樣,是按照屬性創建的時間順序升序排列的。但是Object.key
, for...in
, Object.getOwnPropertyNames
方法不能獲得對象的 Symbol 屬性,Reflect.ownKeys
和 Object.getOwnPropertySymbols
可以。
總結
當一個對象的屬性鍵是上述類型的組合時,該對象的非負整數鍵(可枚舉和不可枚舉)首先按升序添加到數組中,然後按插入順序添加字符串鍵。最後,Symbol 鍵按插入順序加入。
const obj = { 100: 100, 0: 0, a: 'a', [Symbol('a')]: 'a', }; obj[Symbol.for('b')] = 'b'; obj.b = 'b'; console.log(Object.keys(obj)); // [ '0', '100', 'a', 'b' ] console.log(Object.getOwnPropertyNames(obj)); // [ '0', '100', 'a', 'b' ] console.log(Reflect.ownKeys(obj)); // [ '0', '100', 'a', 'b', Symbol(a), Symbol(b) ] for (const key in obj) { console.log('key: ', key); // '0' '100' 'a' 'b' } console.log(Object.getOwnPropertySymbols(obj)); // [ Symbol(a), Symbol(b) ]
但是,如果你強烈依賴插入順序,那麼Map可以保證這一點。
到此這篇關於Javascript中的對象屬性是有序的嗎的文章就介紹到這瞭,更多相關Javascript對象屬性內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- javascript遍歷對象的五種方式實例代碼
- JavaScript遍歷對象的七種方法匯總
- JavaScript循環遍歷的24個方法,你都知道嗎
- 詳解JavaScript的Symbol類型、隱藏屬性、全局註冊表
- 詳解JS對象遍歷的順序問題