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!

推薦閱讀: