
本文介绍一种高效、可扩展的方法,将具有相同数字后缀的键值对(如 `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) 函数。
总结:该方案将「按数字归组」这一通用需求抽象为可配置函数,兼顾简洁性与工程健壮性,适用于表单批量解析、动态表格渲染、配置合并等多种场景。










