使用递归在python中操作json

时间:2017-04-27 19:11:53

标签: python json dictionary recursion

所有

我试图改变一些json看起来的方式,并通过以下方式格式化它:

1. flatten all of the fields lists
2. Then remove the fields lists and replace them with the name : flatten list

示例:

{
"name": "",
"fields": [{
        "name": "keys",
        "fields": [{
                "node-name": "0/0/CPU0"
            },
            {
                "interface-name": "TenGigE0/0/0/47"
            },
            {
                "device-id": "ASR9K-H1902.corp.cisco.com"
            }
        ]
    },
    {
        "name": "content",
        "fields": [{
            "name": "lldp-neighbor",
            "fields": [{
                    "receiving-interface-name": "TenGigE0/0/0/47"
                },
                {
                    "receiving-parent-interface-name": "Bundle-Ether403"
                },
                {
                    "device-id": "ASR9K-H1902.corp.cisco.com"
                },
                {
                    "chassis-id": "78ba.f975.a64f"
                },
                {
                    "port-id-detail": "Te0/1/0/4/0"
                },
                {
                    "header-version": 0
                },
                {
                    "hold-time": 120
                },
                {
                    "enabled-capabilities": "R"
                },
                {
                    "platform": ""
                }
            ]
        }]
    }
]
}

会变成:

{
"": [{

        "keys": [{
            "node-name": "0/0/CPU0",
            "interface-name": "TenGigE0/0/0/47",
            "device-id": "ASR9K-H1902.corp.cisco.com"
        }]
    },
    {
        "content": [{
            "lldp-neighbor": [{
                "receiving-interface-name": "TenGigE0/0/0/47",
                "receiving-parent-interface-name": "Bundle-Ether403",
                "device-id": "ASR9K-H1902.corp.cisco.com",
                "chassis-id": "78ba.f975.a64f",
                "port-id-detail": "Te0/1/0/4/0",
                "header-version": 0,
                "hold-time": 120,
                "enabled-capabilities": "R",
                "platform": ""
            }]
        }]
    }
]
}

我已尝试以下方法将列表展平:

def _flatten_fields(self, fields_list):
        c = {}
        for b in [d for d in fields_list if bool(d)]:
                c.update(b)
        return c

这似乎有效,但我无法找到使用递归进入子级别的方法,我将所有展平列表和名称保存到新词典中,有没有办法通过操作来实现原字典?

2 个答案:

答案 0 :(得分:1)

这适用于您提供的示例:

import json

def flatten(data):
    result = dict()
    if isinstance(data, dict):
        if 'name' in data:
            name = data['name']
            result[name] = flatten(data['fields'])
        else:
            key = data.keys()[0]
            value = data.values()[0]
            result[key] = value
    else:
        for entry in data:
            result.update(flatten(entry))
    return result

print json.dumps(flatten(data), indent=4)

<强>输出

{
    "": {
        "keys": {
            "node-name": "0/0/CPU0", 
            "interface-name": "TenGigE0/0/0/47", 
            "device-id": "ASR9K-H1902.corp.cisco.com"
        }, 
        "content": {
            "lldp-neighbor": {
                "receiving-interface-name": "TenGigE0/0/0/47", 
                "receiving-parent-interface-name": "Bundle-Ether403", 
                "header-version": 0, 
                "port-id-detail": "Te0/1/0/4/0", 
                "chassis-id": "78ba.f975.a64f", 
                "platform": "", 
                "device-id": "ASR9K-H1902.corp.cisco.com", 
                "hold-time": 120, 
                "enabled-capabilities": "R"
            }
        }
    }
}

预期输出中没有显示额外的list图层,但我认为您不需要这些图层。

答案 1 :(得分:0)

这适用于您提供的示例:

def flatten_fields(fields_list):

    c = {}
    for item in fields_list:
        for key in item:
            if key == "fields":
                c[item["name"]] = flatten_fields(item["fields"])
            elif key != "name":
                c[key] = item[key]
            break

    return [c]

但是它适用于字典列表,所以你应该像flatten_fields([data])[0]一样调用它。

输出结果为:

{
"": [{
    "keys": [{
        "node-name": "0/0/CP0", 
        "interface-name": "TenGigE0/0/0/47", 
        "device-id": "ASR9K-H1902.corp.cisco.com"
        }], 
    "content": [{
        "lldp-neighbor": [{
            "chassis-id": "78ba.f975.a64f", 
            "receiving-parent-interface-name": "Bndle-Ether403", 
            "enabled-capabilities": "R", 
            "device-id": "ASR9K-H1902.corp.cisco.com", 
            "hold-time": 120, 
            "receiving-interface-name": "TenGigE0/0/0/47", 
            "platform": "", 
            "header-version": 0, 
            "port-id-detail": "Te0/1/0/4/0"
            }]
        }]
    }]
}