
本文介绍一种通用、可扩展的方法,使用递归函数将形如 `"a.b.c"` 的键名自动解析并构建深层嵌套对象结构,避免硬编码层级,适用于任意深度的路径映射。
在实际开发中(尤其在 IoT 配置解析、JSON Schema 动态生成或前端表单字段扁平化还原等场景),我们常需将一组以点号(.)分隔的扁平键名(如 "0_tags.0.Various.Test-String")及其对应值,动态组装成结构清晰的嵌套对象。原始方案若按固定层级(如 obj[a][b][c])硬编码,不仅难以维护,更无法应对路径长度不一的情况。
下面提供一个健壮、无深度限制的解决方案:
✅ 核心思路:递归路径遍历 + 动态对象创建
通过 split('.') 将每个键拆分为路径数组,再利用递归函数逐层访问/创建对象属性,直至最后一级才赋值。
✅ 完整可运行示例
function createNestedObject(obj, keys, value) {
const [head, ...tail] = keys; // ES6 解构:取首项,剩余为 tail
if (tail.length === 0) {
// 到达叶子节点:直接赋值
obj[head] = value;
} else {
// 中间节点:确保该层级存在,递归进入下一层
if (obj[head] === undefined || typeof obj[head] !== 'object') {
obj[head] = {};
}
createNestedObject(obj[head], tail, value);
}
}
// 示例输入数据(注意:此处 topic 值已按需求差异化处理,实际中可动态替换)
const tagObject = {
"0_tags.0": {
"common": {},
"native": { "topic": "Stromerfassung_251/Zähler0-Leistung" }
},
"0_tags.0.Various": {
"common": {},
"native": { "topic": "Stromerfassung_251/Zähler0-Leistung" }
},
"0_tags.0.Various.Stromerfassung_251/Zähler0-Leistung": {
"common": {},
"native": { "topic": "Stromerfassung_251/Zähler0-Leistung" }
},
"0_tags.0.Various.Test-String": {
"common": {},
"native": { "topic": "Stromerfassung_251/Zähler0-Leistung" }
},
"0_tags.0.Various.battery_charge": {
"common": {},
"native": { "topic": "Stromerfassung_251/Zähler0-Leistung" }
}
};
// 构建结果对象
const result = {};
for (const key in tagObject) {
if (Object.prototype.hasOwnProperty.call(tagObject, key)) {
const keys = key.split('.');
createNestedObject(result, keys, tagObject[key]);
}
}
console.log(JSON.stringify(result, null, 2));⚠️ 注意事项与最佳实践
- 健壮性增强:代码中加入了 hasOwnProperty 检查,避免原型链污染;同时判断中间节点是否为对象,防止意外覆盖(如某键名为 "0_tags.0.toString" 导致错误)。
- 性能考量:对于超大规模数据(>10k 条),可改用迭代版本避免调用栈过深(尾递归优化在 JS 中支持有限,建议显式用 while 循环替代)。
- 路径合法性:若键中含特殊字符(如 . / [ 等),需提前转义或约定编码规则(例如 URL 编码),否则会破坏结构逻辑。
- 值合并策略:当前实现为“后写覆盖”,如需“深度合并”(如 common 对象自动合并而非替换),应引入 lodash.merge 或自定义合并逻辑。
✅ 总结
该方法以简洁的递归模型解耦了路径解析与对象构建,具备完全动态性、强可读性与良好扩展性。无论输入是两层 "a.b" 还是五层 "x.y.z.a.b",均能统一处理——真正实现“一次编写,任意嵌套”。
立即学习“Java免费学习笔记(深入)”;










