Как я могу получить доступ к глубоко вложенному словарю, используя кортежи?

Я хотел бы расширить пример автовивификации, приведенный в предыдущем ответе от nosklo, чтобы разрешить доступ к словарю по кортежу.

Решение nosklo выглядит так:


class AutoVivification(dict):
    """Implementation of perl's autovivification feature."""
    def __getitem__(self, item):
        try:
            return dict.__getitem__(self, item)
        except KeyError:
            value = self[item] = type(self)()
            return value

Тестирование:

a = AutoVivification()

a[1][2][3] = 4
a[1][3][3] = 5
a[1][2]['test'] = 6

print a

Выход:

{1: {2: {'test': 6, 3: 4}, 3: {3: 5}}}

У меня есть случай, когда я хочу установить узел с произвольным набором подписок. Если я не знаю, сколько слоев будет в кортеже, как я могу разработать способ установки соответствующего узла?

Я думаю, что, возможно, я мог бы использовать синтаксис, подобный следующему:

mytuple = (1,2,3)
a[mytuple] = 4

Но у меня возникли проблемы при разработке работающей реализации.


Обновить

У меня есть полностью рабочий пример, основанный на ответе @JCash:

class NestedDict(dict):
    """                                                                       
    Nested dictionary of arbitrary depth with autovivification.               

    Allows data access via extended slice notation.                           
    """
    def __getitem__(self, keys):
        # Let's assume *keys* is a list or tuple.                             
        if not isinstance(keys, basestring):
            try:
                node = self
                for key in keys:
                    node = dict.__getitem__(node, key)
                return node
            except TypeError:
            # *keys* is not a list or tuple.                              
                pass
        try:
            return dict.__getitem__(self, keys)
        except KeyError:
            raise KeyError(keys)
    def __setitem__(self, keys, value):
        # Let's assume *keys* is a list or tuple.                             
        if not isinstance(keys, basestring):
            try:
                node = self
                for key in keys[:-1]:
                    try:
                        node = dict.__getitem__(node, key)
                    except KeyError:
                        node[key] = type(self)()
                        node = node[key]
                return dict.__setitem__(node, keys[-1], value)
            except TypeError:
                # *keys* is not a list or tuple.                              
                pass
        dict.__setitem__(self, keys, value)

Который может получить тот же результат, что и выше, используя расширенную запись среза:

d = NestedDict()
d[1,2,3] = 4
d[1,3,3] = 5
d[1,2,'test'] = 6

1 ответ

Решение

Это похоже на работу

def __setitem__(self, key, value):
    if isinstance(key, tuple):
        node = self
        for i in key[:-1]:
            try:
                node = dict.__getitem__(node, i)
            except KeyError:
                node = node[i] = type(self)()
        return dict.__setitem__(node, i, value)
    return dict.__setitem__(self, key, value)
Другие вопросы по тегам