如何将dict转换为递归defaultdict并加载JSON?

How to convert dict into recursive defaultdict and load JSON?

是否有任何方法可以使用 Python 2.7 将 JSON 代码加载到递归 defaultdict 中以避免 KeyError

例如:

from __future__ import print_function
from collections import defaultdict
import json

S =  '{"a": [{"b": "TEST1", "p": "TEST2"}, {"b": "TEST3", "p": "TEST4"}]}'

d = json.loads(S)
nd = lambda: defaultdict(nd)
ni = nd()
print('1: ', ni['a'][0]['b'])
ni.update(d)
print('2: ', ni['a'][0]['b'])
print('3: ', ni['a'][1]['p'])
print('4: ', ni['a'][1]['c'])

结果

1:  defaultdict(<function <lambda> at 0x0266F530>, {})
2:  TEST1
3:  TEST4
Traceback (most recent call last):
  File "C:/...", line 16, in <module>
    print('4: ', ni['a'][1]['c'])
KeyError: 'c'

ni.update(d)之后ni看起来像dict,而不像递归defaultdict。有什么方法可以将 dict 添加到递归 defaultdict 中并保持其属性?

我希望第 4 步的结果类似于以下内容:

4:  defaultdict(<function <lambda> at 0x0266F530>, {})

仅仅创建一个递归字典是不够的。您必须递归地将 JSON 对象加载到您的字典中。

您的代码中发生的情况是 ni['a'] 是一个普通的字典,而不是 nd 类型的字典。当你 运行 ni.update(d) 时,更新还不够聪明,无法遍历 d,将每个级别的对象写入相似类型的对象。相反,它只是将 d 的第一级值和键复制到 ni 中。一级值就是普通的dicts什么的。

要正确初始化 ni,您需要编写一个递归函数来确定每个值是字典、数组还是标量。如果它是字典,它必须调用 nd,然后用它将以相同递归方式执行的值填充结果。

或者,您可以使用参数 object_hookjson.loads。有关详细信息,请参阅 https://docs.python.org/2/library/json.html#json.load。您在此处传递的任何函数都将在任何解析级别创建的每个字典上调用。所以函数

def defaultdict_from_dict(d):
    nd = lambda: defaultdict(nd)
    ni = nd()
    ni.update(d)
    return ni

或类似的东西可能对你有用。