如何将对象中键名含相同数字后缀的属性配对合并为数组对象

8次阅读

如何将对象中键名含相同数字后缀的属性配对合并为数组对象

本文介绍一种高效、可扩展的方法,将具有相同数字后缀的键值对(如 `lote0` 与 `loteqnt0`)自动分组并组合成结构化对象数组,避免硬编码过滤逻辑,支持任意前缀和多组匹配。

在实际开发中,我们常遇到这类“带编号键名”的扁平对象,例如表单数据、配置项或 API 响应:

const vls = {   "lote0": "jg",   "lote1": "h",   "lote2": "fm",   "loteQnt0": "jgvalue",   "loteQnt1": "hvalue",   "loteQnt2": "fmvalue" };

目标是按数字后缀(0, 1, 2…)将匹配的键聚合,生成如下结构的数组:

[   { name: "jg", value: "jgvalue" },   { name: "h",  value: "hvalue"  },   { name: "fm", value: "fmvalue" } ]

推荐实现(健壮、可复用、无硬编码):
我们不预先假设前缀(如 “lote” 或 “loteQnt”),而是通过正则提取所有键的「数字后缀」,再按该数字分组键名,最后映射为统一字段名的对象。

function groupByNumberSuffix(obj, keyMap = {}) {   // 步骤 1:提取所有键的数字后缀,并归类   const groups = new Map(); // Map    Object.entries(obj).forEach(([key, value]) => {     const match = key.match(/(D+)(d+)$/); // 匹配末尾数字,如 'lote0' → ['lote0','lote','0']     if (!match) return;      const [, prefix, numStr] = match;     const num = Number(numStr);      if (!groups.has(num)) {       groups.set(num, {});     }     groups.get(num)[prefix] = value;   });    // 步骤 2:按 keyMap 映射字段名(默认:prefix → 小驼峰首字母)   const result = [];   for (const [num, groupObj] of groups.entries()) {     const mapped = {};     for (const [prefix, val] of Object.entries(groupObj)) {       const targetKey = keyMap[prefix] ||                          prefix.replace(/^[a-z]/, c => c.toUpperCase()); // 如 'lote' → 'Lote'       mapped[targetKey] = val;     }     result.push(mapped);   }    // 步骤 3:按数字升序返回(确保 0,1,2…顺序)   return result.sort((a, b) => {     const numA = Object.keys(a).find(k => /d+/.test(k)) || '0';     const numB = Object.keys(b).find(k => /d+/.test(k)) || '0';     return parseInt(numA) - parseInt(numB);   }); }  // 使用示例: const vls = {   "lote0": "jg", "lote1": "h", "lote2": "fm",   "loteQnt0": "jgvalue", "loteQnt1": "hvalue", "loteQnt2": "fmvalue" };  const result = groupByNumberSuffix(vls, {   lote: 'name',   loteQnt: 'value' });  console.log(result); // → [ //     { name: "jg", value: "jgvalue" }, //     { name: "h",  value: "hvalue"  }, //     { name: "fm", value: "fmvalue" } //   ]

? 关键优势说明:

  • 动态识别后缀:无需手动拆分 lote* 和 loteQnt*,正则 /(D+)(d+)$/ 自动捕获前缀与数字;
  • 灵活字段映射:通过 keyMap 参数声明 lote → name、loteQnt → value,语义清晰且可扩展;
  • 健壮容错:跳过无数字后缀的键(如 “meta”),不中断流程;
  • 有序输出:结果按数字升序排列,符合直觉;
  • 零依赖:纯 javaScript,兼容 ES2015+。

⚠️ 注意事项:

  • 键名必须以「非数字字符 + 数字」结尾(如 item123 ✅,123item ❌,item_123 ❌);若需支持下划线等分隔符,可调整正则为 /(D+?)_(d+)$/ 或 /(D+?)(d+)$/;
  • 若同一数字对应多个同前缀键(如两个 lote0),后者会覆盖前者——建议确保原始数据键唯一;
  • 如需支持嵌套结构或更复杂匹配逻辑,可进一步封装为 groupByPattern(patterns) 函数。

总结:该方案将「按数字归组」这一通用需求抽象为可配置函数,兼顾简洁性与工程健壮性,适用于表单批量解析、动态表格渲染、配置合并等多种场景。

text=ZqhQzanResources