JS 解惑 - Object 中的 key 是有序的么?

发布于 2023-08-11 05:14:51 字数 2794 浏览 34 评论 0

当我们使用 for/in 遍历一个 Object 对象的时候,打印的结果是否按 key 的顺序打印出来呢?

答案是: 不一定

背景

最近在做一个项目的时候,遇到这样一个需求:一个下拉列表中有 3 个固定选项,包括: -1:全部;0:正常;1:失效 。于是,我就定义了一个对象,然后循环这个对象,把结果放到 <option> 上面:

var obj = {
    '-1': '全部',
    '0' : '正常',
    '1' : '失效'
};
<!-- Vue 代码片段 -->
<select>
    <option v-for="(item, key) in obj" :value="key">{{item}}</option>
</select>

<!-- 结果却是: -->
<!--  0  正常 -->
<!--  1  失效 -->
<!--  -1  全部 -->

于是就有了今天这篇文章,且看下文。

解惑

Object 的 key 的排序规则到底是什么样子的呢?答案是:

如果 key 是整数(如:123)或者整数类型的字符串(如:“123”),那么会按照从小到大的排序。除此之外,其它数据类型,都安装对象 key 的实际创建顺序排序。

var obj = {
    '-1': '全部',
    '0' : '正常',
    '1' : '失效'
};
for (let key in obj) {
     console.log(key, obj[key]);
};
// result
// 0 正常
// 1 失效
// -1 全部

另外,如果 key 中除了整数或者整数类型的字符串外,还含有其它数据类型,则整数放在最前面,比如:

var obj = {
    'a': 111,
    '我' : 222,
    '1' : 333,
    '1.3': 444,
    '3': 555
};
for (let key in obj) {
     console.log(key, obj[key]);
};
// result
// 1 333
// 3 555
// a 111
// 我 222
// 1.3 444

解决

那还是上面的问题,我如何让对象按 key 的顺序输出呢?答案是:

将 key 转换成非整数类型的字符串,使用的时候再还原。

如果全部是类整数的 key,则可以这么做:

// 每个 key 后面加.转换成字符串
var obj = {
    '-1.': '全部',
    '0.' : '正常',
    '1.' : '失效'
};
for (let key in obj) {
    // ~~ 表示转换成整数,这样上面的 key 又还原成了-1/0/1
    console.log(~~key, obj[key]);
};
// result
// -1 全部
// 0 正常
// 1 失效

但是,如果 key 是由各种数据类型混合的,那就不能转换成整数了,可以这么做:

// 每个 key 前面加.转换成字符串
var obj = {
    '.a': 111,
    '.我' : 222,
    '.1' : 333,
    '.1.3': 444,
    '.3': 555
};
for (let key in obj) {
    // 从第 1 个字符取原始的 key
    console.log(key.substring(1), obj[key]);
};
// result
// a 111
// 我 222
// 1 333
// 1.3 444
// 3 555

最后

回归到我最初遇到的问题,那就这么解决了:

var obj = {
    '-1.': '全部',
    '0.' : '正常',
    '1.' : '失效'
};
<select>
    <option v-for="(item, key) in obj" :value="~~key">{{item}}</option>
</select>

补充方案 2019 年 06 月 27 日 11:24:47

最近又遇到一个问题,在详情页面后台返回 key,但是页面上要展示文案,所以按照上面 “最后” 的解决方案,我们就只能这么做了:

{{obj[`${data.key}.`]}}

也就是取值显示的时候,需要 key+. 来拼接,这个写法就非常不友好了。

所以,最后我们采用了 Map 来处理这个事情,处理上基本算完美了,方案如下:

定义对象:

const obj = new Map([
    [-1, '全部'],
    [0, '正常'],
    [1, '失效']
]);

循环遍历时:

<select>
    <option v-for="[key, item] in Array.from(obj)" :value="key">{{item}}</option>
</select>

详情查看时:

{{obj.get(data.key)}}

参考

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

我早已燃尽

暂无简介

文章
评论
26 人气
更多

推荐作者

櫻之舞

文章 0 评论 0

弥枳

文章 0 评论 0

m2429

文章 0 评论 0

野却迷人

文章 0 评论 0

我怀念的。

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文