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

本文介绍一种高效、可扩展的方法,将具有相同数字后缀的键值对(如 `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),后者会覆盖前者——建议确保原始数据键唯一;
  • 如需支持嵌套结构或更复杂匹配逻辑,可进一步封装为 grou

    pByPattern(patterns) 函数。

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