猿问

有效地将 JavaScript 键和值数组转换为对象

我需要将两个 JavaScript 数组转换为对象列表。其中一个输入数组表示输出对象的键,另一个包含其值(以及一些其他信息,与此问题无关)。


示例数据:


let theKeys = ['firstName', 'lastName', 'city'];

let theValues = [{data: [['John', 'Smith', 'New York'],

                         ['Mike', 'Doe', 'Chicago'],

                         ...

                        ],

                 otherStuff: ...}

                ];

上述所需的输出:


output = [{

            firstName: 'John',

            lastName: 'Smith',

            city: 'New York'

          },

          {

            firstName: 'Mike',

            lastName: 'Doe',

            city: 'Chicago',

          },

          ...

         ]

(这只是一个示例,我的实际数据来自 REST 响应,内容和长度可能有所不同。我正在使用显示表格数据的 Vue 应用程序。)


下面我现有的代码适用于少量数据,但在处理大量数据时会使所有浏览器崩溃或挂起。


return this.theValues.flatMap(results => {

  let jsonified = [];


  for (let v = 0; v < results.theValues.length; v++) {

    let singleJson = {}; 


    for (let k = 0; k < this.theKeys.length; k++) {

      let key = this.theKeys[k];

      singleJson[key] = results.data[v][k];

    }


    jsonified.push(singleJson);

  }


  return jsonified;

});

对于少至几千个结果,运行需要几分钟。我怎样才能让它更快?我是否缺少一些操作可以让我避免嵌套 for 循环?


汪汪一只猫
浏览 139回答 3
3回答

斯蒂芬大帝

最简单的方法可能是将.map值放入键值元组并调用Object.fromEntries它:const theKeys = ['firstName', 'lastName', 'city'];const theValues = [{&nbsp; data: [&nbsp; &nbsp; ['John', 'Smith', 'New York'],&nbsp; &nbsp; ['Mike', 'Doe', 'Chicago']&nbsp; ]}];console.log(&nbsp; theValues[0].data.map(e =>&nbsp; &nbsp; Object.fromEntries(e.map((e,i) => [theKeys[i], e]))&nbsp; ))

慕雪6442864

如果您对属性进行硬编码而不是查看,您可以摆脱内部循环theKeys,但我怀疑您会想要这样。您唯一真正不需要的是flatMap. 无论如何,大多数通用数组方法的速度并不为人所知(例如,forEach通常比普通for循环慢)。FWIW,这似乎表现良好:let result = [];for (let i = 0; i < theValues[0].data.length; i++) {&nbsp; &nbsp; let resultObj = {};&nbsp; &nbsp; for (let j = 0; j < theKeys.length; j++) {&nbsp; &nbsp; &nbsp; &nbsp; resultObj[theKeys[j]] = theValues[0].data[i][j];&nbsp; &nbsp; }&nbsp; &nbsp; result.push(resultObj);}我用 11k 个项目对其进行了测试,在 Chrome 中运行时间约为 5 毫秒。对于 90k 项,仍然只需要大约 30 毫秒。

FFIVE

您可以放弃这flatMap应该节省一些性能,只需使用普通循环完成所有操作:const result = []for (const v of theValues) {&nbsp; &nbsp; for (const entry of v.data) {&nbsp; &nbsp; &nbsp; &nbsp; const obj = {}&nbsp; &nbsp; &nbsp; &nbsp; for (let i = 0; i < entry.length; i++) {&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; obj[theKeys[i]] = entry[i]&nbsp; &nbsp; &nbsp; &nbsp; }&nbsp; &nbsp; &nbsp; &nbsp; result.push(obj)&nbsp; &nbsp; }}编辑:微观优化const result = []const keysLength = theKeys.lengthfor (let i = theValues.length - 1; i >= 0; i--) {&nbsp; &nbsp; const data = theValues[i].data&nbsp; &nbsp; for (let j = data.length - 1; j >= 0; j--) {&nbsp; &nbsp; &nbsp; &nbsp; const entry = data[j]&nbsp; &nbsp; &nbsp; &nbsp; const obj = {}&nbsp; &nbsp; &nbsp; &nbsp; for (let k = keysLength - 1; k >= 0; k--) {&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; obj[theKeys[k]] = entry[k]&nbsp; &nbsp; &nbsp; &nbsp; }&nbsp; &nbsp; &nbsp; &nbsp; result.push(obj)&nbsp; &nbsp; }}
随时随地看视频慕课网APP

相关分类

JavaScript
我要回答