返回介绍

成员关系:__contains__、__iter__ 和 __getitem__

发布于 2024-01-29 22:24:15 字数 1536 浏览 0 评论 0 收藏 0

迭代器的内容比我们目前所见到的还要丰富。运算符重载往往是多个层级的:类可以提供特定的方法,或者用作退而求其次选项的更通用的替代方案。例如:

·Python 2.6中的比较使用__lt__这样的特殊方法来表示少于比较(如果有的话),或者使用通用的__cmp__。Python 3.0只使用特殊的方法,而不是__cmp__,如本章后面所介绍的。

·布尔测试类似于先尝试一个特定的__bool__(以给出一个明确的True/False结果),并且,如果没有它,将会退而求其次到更通用的__len__(一个非零的长度意味着True)。正如我们将在本章随后见到的,Python 2.6也一样起作用,但是,使用名称__nonzero__而不是__bool__。

在迭代领域,类通常把in成员关系运算符实现为一个迭代,使用__iter__方法或__getitem__方法。要支持更加特定的成员关系,类可能编写一个__contains__方法——当出现的时候,该方法优先于__iter__方法,__iter__方法优先于__getitem__方法。__contains__方法应该把成员关系定义为对一个映射应用键(并且可以使用快速查找),以及用于序列的搜索。

考虑如下的类,它编写了所有3个方法和测试成员关系以及应用于一个实例的各种迭代环境。调用的时候,其方法会打印出跟踪消息:

这段脚本运行的时候,其输出如下所示——特定的__contains__拦截成员关系,通用的__iter__捕获其他的迭代环境以至__next__重复地被调用,而__getitem__不会被调用:

但是,要观察如果注释掉__contains__方法后代码的输出发生了什么变化——成员关系现在路由到了通用的__iter__:

最后,如果__contains__和__iter__都注释掉的话,其输出如下——索引__getitem__替代方法会被调用,针对成员关系和其他迭代环境使用连续较高的索引:

正如我们所看到的,__getitem__方法甚至更加通用:除了迭代,它还拦截显式索引和分片。分片表达式用包含边界的一个分片对象来触发__getitem__,既针对内置类型,也针对用户定义的类,因此,我们的类中分片是自动化的:

然而,在并非面向序列的、更加现实的迭代用例中,__iter__方法可能很容易编写,因为它不必管理一个整数索引,并且__contains__考虑到作为一种特殊情况优化成员关系。

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文