前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >JavaScript对象属性是有序的吗?

JavaScript对象属性是有序的吗?

作者头像
前端小智@大迁世界
发布2022-09-08 12:18:22
1.5K0
发布2022-09-08 12:18:22
举报
文章被收录于专栏:终身学习者

有梦想,有干货,微信搜索 【大迁世界】 关注这个在凌晨还在刷碗的刷碗智。 本文 GitHub https://github.com/qq449245884/xiaozhi 已收录,有一线大厂面试完整考点、资料以及我的系列文章。

最近有人问我,JavaScript对象属性是否一定是无序的、不可预测的?

早期接触过JavaScript的开发者可能会回答,Object.keys()for...in会返回一个不可预知的对象属性顺序。

但现在的情况仍然是这样吗?

不是了,有些情况下是有序的。

从ECMAScript 2020开始,Object.keyfor...inObject.getOwnPropertyNamesReflect.ownKeys都遵循同一个规范顺序。它们是:

1. 自己的属性是数组的索引,按数字索引升序排列

代码语言:javascript
复制
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)
}
代码语言:javascript
复制
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 中。

3. 自身的 Symbol 属性,按创建时间顺序递增

代码语言:javascript
复制
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.ownKeysObject.getOwnPropertySymbols 可以。

总结

当一个对象的属性键是上述类型的组合时,该对象的非负整数键(可枚举和不可枚举)首先按升序添加到数组中,然后按插入顺序添加字符串键。最后,Symbol 键按插入顺序加入。

代码语言:javascript
复制
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可以保证这一点。


编辑中可能存在的bug没法实时知道,事后为了解决这些bug,花了大量的时间进行log 调试,这边顺便给大家推荐一个好用的BUG监控工具 Fundebug

作者:islizeqiang 译者:小智 来源:medium

原文:https://medium.com/@islizeqia...

交流

有梦想,有干货,微信搜索 【大迁世界】 关注这个在凌晨还在刷碗的刷碗智。 本文 GitHub https://github.com/qq449245884/xiaozhi 已收录,有一线大厂面试完整考点、资料以及我的系列文章。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2022-07-16,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 1. 自己的属性是数组的索引,按数字索引升序排列
  • 3. 自身的 Symbol 属性,按创建时间顺序递增
  • 总结
    • 交流
    领券
    问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档