面试官:JavaScript对象属性是有序的吗?

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

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

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

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

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

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

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

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 中。

2. 自身的 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.keyfor...in,Object.getOwnPropertyNames方法不能获得对象的 Symbol 属性,Reflect.ownKeysObject.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可以保证这一点。

作者:islizeqiang

译者:小智

来源:medium

原文:https://medium.com/@
islizeqiang/are-javascript-object-properties-ordered-c30597754e5c

责任编辑:武晓燕 来源: 今日头条
相关推荐

2023-11-27 08:32:02

元素HashMap

2022-02-08 08:14:07

Context数据线程

2023-08-11 17:13:39

JavaScrip

2023-08-13 16:17:31

2021-09-07 10:44:33

Java 注解开发

2020-03-10 08:01:05

Java堆内存线程共享

2022-07-18 13:59:43

Redis单线程进程

2021-02-19 10:02:57

HTTPSJava安全

2024-02-04 10:08:34

2022-09-29 07:30:57

数据库索引字段

2022-08-02 06:31:32

Java并发工具类

2022-07-26 08:40:42

Java并发工具类

2024-02-22 15:36:23

Java内存模型线程

2021-12-08 06:53:29

面试动态代理

2015-08-13 10:29:12

面试面试官

2021-03-16 07:13:07

Java对象存储

2023-02-16 07:30:38

引用计数算法

2021-05-20 08:54:16

Go面向对象

2023-02-08 07:04:20

死锁面试官单元

2021-09-08 08:06:57

Redis原子性数据类型
点赞
收藏

51CTO技术栈公众号