JS 解惑 - Object 中的 key 是有序的么?
当我们使用 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 技术交流群。

绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论