JavaScript对象属性是否一定是无序的、不可预测的?
早期接触过JavaScript的开发者可能会回答,Object.keys()
或for...in
会返回一个不可预知的对象属性顺序。
但现在的情况仍然是这样吗?
不是了,有些情况下是有序的。
从ECMAScript 2020开始,Object.key
、for...in
、Object.getOwnPropertyNames
和Reflect.ownKeys
都遵循同一个规范顺序。它们是:
1. 自己的属性是数组的索引,按数字索引升序排列
上面的代码添加了事件循环的知识点。因为 setTimeout
是一个异步的宏任务,当console.log
输出时,c
属性还没有被添加到 obj
中。
2. 自身的 Symbol 属性,按创建时间顺序递增
Symbol 属性和 String 属性一样,是按照属性创建的时间顺序升序排列的。但是Object.key
, for...in
, Object.getOwnPropertyNames
方法不能获得对象的 Symbol 属性,Reflect.ownKeys
和 Object.getOwnPropertySymbols
可以。
总结
当一个对象的属性键是上述类型的组合时,该对象的非负整数键(可枚举和不可枚举)首先按升序添加到数组中,然后按插入顺序添加字符串键。最后,Symbol 键按插入顺序加入。
但是,如果你强烈依赖插入顺序,那么Map可以保证这一点。