本文介绍一种通用、可扩展的方法,使用递归函数将形如 `"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));该方法以简洁的递归模型解耦了路径解析与对象构建,具备完全动态性、强可读性与良好扩展性。无论输入是两层 "a.b" 还是五层 "x.y.z.a.b",均能统一处理——真正实现“一次编写,任意嵌套”。
立即学习“Java免费学习笔记(深入)”;