• 4.2 代理迭代
    • 问题
    • 解决方案
    • 讨论

    4.2 代理迭代

    问题

    你构建了一个自定义容器对象,里面包含有列表、元组或其他可迭代对象。你想直接在你的这个新容器对象上执行迭代操作。

    解决方案

    实际上你只需要定义一个 iter() 方法,将迭代操作代理到容器内部的对象上去。比如:

    1. class Node:
    2. def __init__(self, value):
    3. self._value = value
    4. self._children = []
    5.  
    6. def __repr__(self):
    7. return 'Node({!r})'.format(self._value)
    8.  
    9. def add_child(self, node):
    10. self._children.append(node)
    11.  
    12. def __iter__(self):
    13. return iter(self._children)
    14.  
    15. # Example
    16. if __name__ == '__main__':
    17. root = Node(0)
    18. child1 = Node(1)
    19. child2 = Node(2)
    20. root.add_child(child1)
    21. root.add_child(child2)
    22. # Outputs Node(1), Node(2)
    23. for ch in root:
    24. print(ch)

    在上面代码中, iter() 方法只是简单的将迭代请求传递给内部的 _children 属性。

    讨论

    Python的迭代器协议需要 iter() 方法返回一个实现了 next() 方法的迭代器对象。如果你只是迭代遍历其他容器的内容,你无须担心底层是怎样实现的。你所要做的只是传递迭代请求既可。

    这里的 iter() 函数的使用简化了代码,iter(s) 只是简单的通过调用 s.iter() 方法来返回对应的迭代器对象,就跟 len(s) 会调用 s.len() 原理是一样的。

    原文:

    http://python3-cookbook.readthedocs.io/zh_CN/latest/c04/p02_delegating_iteration.html