迭代嵌套字典

时间:2012-11-14 16:41:03

标签: python dictionary trie

我想为我的'玩具'Trie实现编写一个迭代器。

添加已经如下:

class Trie:
    def __init__(self):
        self.root = dict()
        pass
    def add(self, string, value):
        global nops
        current_dict = self.root
        for letter in string:
           nops += 1
           current_dict = current_dict.setdefault(letter, {})
        current_dict = current_dict.setdefault('value', value)              
        pass

添加的输出如下:

trie = Trie()
trie.add("hello",1)
trie.add("world",2)
trie.add("worlds",12)
print trie.root
{'h': {'e': {'l': {'l': {'o': {'value': 1}}}}}, 'w': {'o': {'r': {'l': {'d': {'s': {'value': 12}, 'value': 2}}}}}}

我知道,我需要a __iter__ and next method

def __iter__(self):
    self.root.__iter__()
    pass

 def next(self):
    print self.root.next()

但是AttributeError: 'dict' object has no attribute 'next'。我该怎么办?

[更新]在完美的世界中,我希望输出为一个dict,其中所有单词/条目及其对应的值。

2 个答案:

答案 0 :(得分:2)

您的__iter__特殊方法应返回迭代器;也就是说,您可以调用next的类的对象。玩具迭代器类就像:

class MyIterator:
    def __init__(self):
        self.i = 10
    def next(self):
        self.i -= 1
        if self.i == 0:
            raise StopIteration
        else:
            return self.i

除非你有一个自然对象可以打开iter,否则通常更容易让__iter__成为生成器

def __iter__(self):
    for i in range(10)
        yield i

这是Trie的基于堆栈的生成器迭代器:

    def __iter__(self):
        stack = [('', self.root)]
        while stack:
            prefix, d = stack.pop()
            for k, v in d.items():
                if k == 'value':
                    yield prefix, v
                else:
                    stack.append((prefix + k, v))

您也可以尝试递归编写它,尽管您需要使用itertools.chainyield from(仅限Python 3.3)。

答案 1 :(得分:0)

假设我正确理解你在迭代器中寻找的东西,我会沿着这些方向做一些事情(注意__init__中添加的字段:

class Trie:
    def __init__(self):
        self.root = dict()
        self.stack = ''

    def add(self, string, value):
        """map it to the given value."""
        global nops
        current_dict = self.root
        for letter in string:
           nops += 1
           current_dict = current_dict.setdefault(letter, {})
        current_dict = current_dict.setdefault('value', value)              
        pass

    def __iter__(self, input_dict=None):
        if not input_dict:
            input_dict = self.root

        if 'value' in input_dict:
            yield (self.stack, input_dict['value'])

        keys = [x for x in input_dict.keys() if x != 'value']
        for key in keys:
            self.stack += key
            for item in self.__iter__(input_dict[key]):
                yield item
            self.stack = self.stack[:-1]

那会给你:

trie = Trie()
trie.add("hello", 1)
trie.add("world", 2)
trie.add("worlds", 12)
print trie.root
print [x for x in trie]

# {'h': {'e': {'l': {'l': {'o': {'value': 1}}}}}, 'w': {'o': {'r': {'l': {'d': {'s': {'value': 12}, 'value': 2}}}}}}
# [('hello', 1), ('world', 2), ('worlds', 12)]

因此,可以在没有itertools.chainyield from的情况下递归执行此操作。