代码之家  ›  专栏  ›  技术社区  ›  TJB

在Python中处理深度嵌套字典的便捷方法

  •  7
  • TJB  · 技术社区  · 6 年前

    我有一本python中嵌套很深的词典,它占用了很多空间。有没有办法缩写这样的东西

    master_dictionary['sub_categories'][sub_cat_name]['attributes'][attribute_name]['special_type']['nested_children'][child_cat_name][color] = blue
    

    以这个为例

    nested_child_info[color] = blue
    

    还让它编辑字典?我希望这是有意义的。

    7 回复  |  直到 6 年前
        1
  •  6
  •   wwii    6 年前

    类似于@fferri。您必须始终指定 长的 列表获取 参考 使用 reduce getitem :

    from functools import reduce
    from operator import getitem
    
    d = {1:{2:{3:{4:5}}}}
    
    foo = 2
    items = [1,foo,3]
    result = d
    info = reduce(getitem, items, d)
    
    
    >>> info[4]
    5
    >>> d
    {1: {2: {3: {4: 5}}}}
    >>> info[4] = 99
    >>> d
    {1: {2: {3: {4: 99}}}}
    

    我也在玩弄一个类,但它似乎没有太多的优势——除了您可以自定义一个键错误异常,以便错误消息可以告诉您缺少哪个深度的键。

    class Drilldown:
        def __init__(self, d, path):
            #self.final = reduce(getitem, path, d)
            self.final = d
            for i, item in enumerate(path, 1):
                try:
                    self.final = self.final[item]
                except KeyError as e:
                    msg = ''.join('[{}]' for _ in range(i))
                    msg = msg.format(*path[:i])
                    msg = 'The last key in the path "{}" does not exist'.format(msg)
                    e.args = [msg]
                    raise
        def __call__(self, item):
            return self.final[item]
        def __setitem__(self, item, value):
            self.final[item] = value
        def __getitem__(self, item):
            return self.final[item]
        def __str__(self):
            return str(self.final)
        def __repr__(self):
            return repr(self.final)
    
    >>> z = 19
    >>> items = [1,2,z]
    >>> q = Drilldown(d,items)
    Traceback (most recent call last):
      File "<pyshell#68>", line 1, in <module>
        q = Drilldown(d,items)
      File "C:\pyProjects33\tmp.py", line 32, in __init__
        self.final = self.final[item]
    KeyError: 'The last key in the path "[1][2][19]" does not exist'
    
    >>> 
    >>> #normal usage
    >>> items = [1,2,3]
    >>> q = Drilldown(d,items)
    >>> d
    {1: {2: {3: {4: 5}}}}
    >>> q
    {4: 5}
    >>> q(4)
    5
    >>> q[4]
    5
    >>> q[4] += 20
    >>> q
    {4: 25}
    >>> d
    {1: {2: {3: {4: 25}}}}
    >>> q['foo'] = '99'
    >>> q
    {4: 25, 'foo': '99'}
    >>> d
    {1: {2: {3: {4: 25, 'foo': '99'}}}}
    >>> 
    
        2
  •  4
  •   fferri    6 年前
    nested_child_info = master_dictionary['sub_categories'][sub_cat_name]['attributes'][attribute_name]['special_type']['nested_children'][child_cat_name]
    nested_child_info[color] = blue
    

    nested_child_info 是一个 参考 ,因此更改其内容将更改 master_dictionary

        3
  •  3
  •   Vamsi cdrom    6 年前

    是的,你可以。

    >>> dict1 = {'foo':{'bar':{'baz':0}}}
    >>> dict2 = dict1['foo']['bar']
    >>> dict2['baz'] = 1
    >>> dict1
    {'foo': {'bar': {'baz': 1}}} # dict1 has been modified
    
        4
  •  2
  •   Colin Ricardo    6 年前

    你能这样做吗?

    thing = {1: {2: {3: {4: {'hello': 'world'}}}}}
    a = thing[1]
    b = a[2]
    c = b[3]
    d = c[4]
    print(d) # {'hello': 'world'}
    
        5
  •  2
  •   MrName    6 年前

    由于字典是可变的,因此这实际上会完全按照您的预期发生:

    >>> test = {'outer': 'thing', 'inner': {'thing': 'im the inner thing'}}
    >>> inner_thing = test['inner']
    >>> inner_thing
    {'thing': 'im the inner thing'}
    >>> inner_thing['thing'] = 'im something new'
    >>> inner_thing
    {'thing': 'im something new'}
    >>> test
    {'outer': 'thing', 'inner': {'thing': 'im something new'}}
    

    这是因为在python中可变对象是通过引用传递的,而不是作为副本传递的(关于这方面有很多很好的文章,所以不打算详述)。

    然而,值得注意的是,您可能实际上并不想更改原始dict,因为它可能会对使用此变量的其他代码产生不希望的影响(这取决于您的代码库)。在这种情况下,我通常会复制需要变异的数据,以避免副作用:

    >>> from copy import deepcopy
    >>> test = {'outer': 'thing', 'inner': {'thing': 'im the inner thing'}}
    >>> new_test = deepcopy(test)
    >>> inner_thing = test['inner']
    >>> inner_thing['thing'] = 'im something new'
    >>> test
    {'outer': 'thing', 'inner': {'thing': 'im something new'}}
    >>> new_test
    {'outer': 'thing', 'inner': {'thing': 'im the inner thing'}}
    
        6
  •  2
  •   nucsit026 Bill Letson    5 年前

    如果您有一些“固定”键,则始终可以创建一个函数:

    考虑以下示例:

    d = dict(a=dict(sub_categories=dict(b=1)))
    
    def changevalue(value, lvl1, lvl2):
        d[lvl1]['sub_categories'][lvl2] = value
    
    changevalue(2,'a','b')
    
    print(d)
    
    #{'a': {'sub_categories': {'b': 2}}}
    

    在您的情况下,您可能需要:

    [sub_cat_name] ,则, [attribute_name] ,则, [child_cat_name] ,则, [color] 。。。大概

        7
  •  0
  •   edd313    3 年前

    您可以考虑使用 NestedDict .这是一个比较

    # dictionary
    master_dictionary['sub_categories'][sub_cat_name]['attributes'][attribute_name]['special_type']['nested_children'][child_cat_name][color] = blue
    
    # NestedDict
    key = (
        'sub_categories', 
        sub_cat_name, 
        'attributes', 
        attribute_name, 
        'special_type', 
        'nested_children', 
        child_cat_name, 
        color
    )
    master_dictionary[key] = blue
    

    你可以找到 NestedDict公司 在里面 ndicts ,在PyPi上

    pip install ndicts