- 前言
- 目标读者
- 非目标读者
- 本书的结构
- 以实践为基础
- 硬件
- 杂谈:个人的一点看法
- Python 术语表
- Python 版本表
- 排版约定
- 使用代码示例
- 第一部分 序幕
- 第 1 章 Python 数据模型
- 第二部分 数据结构
- 第 2 章 序列构成的数组
- 第 3 章 字典和集合
- 第 4 章 文本和字节序列
- 第三部分 把函数视作对象
- 第 5 章 一等函数
- 第 6 章 使用一等函数实现设计模式
- 第 7 章 函数装饰器和闭包
- 第四部分 面向对象惯用法
- 第 8 章 对象引用、可变性和垃圾回收
- 第 9 章 符合 Python 风格的对象
- 第 10 章 序列的修改、散列和切片
- 第 11 章 接口:从协议到抽象基类
- 第 12 章 继承的优缺点
- 第 13 章 正确重载运算符
- 第五部分 控制流程
- 第 14 章 可迭代的对象、迭代器和生成器
- 14.1 Sentence 类第1版:单词序列
- 14.2 可迭代的对象与迭代器的对比
- 14.3 Sentence 类第2版:典型的迭代器
- 14.4 Sentence 类第3版:生成器函数
- 14.5 Sentence 类第4版:惰性实现
- 14.6 Sentence 类第5版:生成器表达式
- 14.7 何时使用生成器表达式
- 14.8 另一个示例:等差数列生成器
- 14.9 标准库中的生成器函数
- 14.10 Python 3.3 中新出现的句法:yield from
- 14.11 可迭代的归约函数
- 14.12 深入分析 iter 函数
- 14.13 案例分析:在数据库转换工具中使用生成器
- 14.14 把生成器当成协程
- 14.15 本章小结
- 14.16 延伸阅读
- 第 15 章 上下文管理器和 else 块
- 第 16 章 协程
- 第 17 章 使用期物处理并发
- 第 18 章 使用 asyncio 包处理并发
- 第六部分 元编程
- 第 19 章 动态属性和特性
- 第 20 章 属性描述符
- 第 21 章 类元编程
- 结语
- 延伸阅读
- 附录 A 辅助脚本
- Python 术语表
- 作者简介
- 关于封面
11.8 Tombola 子类的测试方法
我编写的 Tombola 示例测试脚本用到两个类属性,用它们内省类的继承关系。
__subclasses__()
这个方法返回类的直接子类列表,不含虚拟子类。
_abc_registry
只有抽象基类有这个数据属性,其值是一个 WeakSet 对象,即抽象类注册的虚拟子类的弱引用。
为了测试 Tombola 的所有子类,我编写的脚本迭代 Tombola.__subclasses__() 和 Tombola._abc_registry 得到的列表,然后把各个类赋值给在 doctest 中使用的 ConcreteTombola。
这个测试脚本成功运行时输出的结果如下:
$ python3 tombola_runner.py BingoCage 24 tests, 0 failed - OK LotteryBlower 24 tests, 0 failed - OK TumblingDrum 24 tests, 0 failed - OK TomboList 24 tests, 0 failed - OK
测试脚本的代码在示例 11-15 中,doctest 在示例 11-16 中。
示例 11-15 tombola_runner.py:Tombola 子类的测试运行程序
import doctest from tombola import Tombola # 要测试的模块 import bingo, lotto, tombolist, drum ➊ TEST_FILE = 'tombola_tests.rst' TEST_MSG = '{0:16} {1.attempted:2} tests, {1.failed:2} failed - {2}' def main(argv): verbose = '-v' in argv real_subclasses = Tombola.__subclasses__() ➋ virtual_subclasses = list(Tombola._abc_registry) ➌ for cls in real_subclasses + virtual_subclasses: ➍ test(cls, verbose) def test(cls, verbose=False): res = doctest.testfile( TEST_FILE, globs={'ConcreteTombola': cls}, ➎ verbose=verbose, optionflags=doctest.REPORT_ONLY_FIRST_FAILURE) tag = 'FAIL' if res.failed else 'OK' print(TEST_MSG.format(cls.__name__, res, tag)) ➏ if __name__ == '__main__': import sys main(sys.argv)
❶ 导入包含 Tombola 真实子类和虚拟子类的模块,用于测试。
❷ __subclasses__() 返回的列表是内存中存在的直接子代。即便源码中用不到想测试的模块,也要将其导入,因为要把那些类载入内存。
❸ 把 _abc_registry(WeakSet 对象)转换成列表,这样方能与 __subclasses__() 的结果拼接起来。
❹ 迭代找到的各个子类,分别传给 test 函数。
❺ 把 cls 参数(要测试的类)绑定到全局命名空间里的 ConcreteTombola 名称上,供 doctest 使用。
❻ 输出测试结果,包含类的名称、尝试运行的测试数量、失败的测试数量,以及 'OK' 或 'FAIL' 标记。
doctest 文件如示例 11-16 所示。
示例 11-16 tombola_tests.rst:Tombola 子类的 doctest
============== Tombola tests ============== Every concrete subclass of Tombola should pass these tests. Create and load instance from iterable:: >>> balls = list(range(3)) >>> globe = ConcreteTombola(balls) >>> globe.loaded() True >>> globe.inspect() (0, 1, 2) Pick and collect balls:: >>> picks = [] >>> picks.append(globe.pick()) >>> picks.append(globe.pick()) >>> picks.append(globe.pick()) Check state and results:: >>> globe.loaded() False >>> sorted(picks) == balls True Reload:: >>> globe.load(balls) >>> globe.loaded() True >>> picks = [globe.pick() for i in balls] >>> globe.loaded() False Check that `LookupError` (or a subclass) is the exception thrown when the device is empty:: >>> globe = ConcreteTombola([]) >>> try: ... globe.pick() ... except LookupError as exc: ... print('OK') OK Load and pick 100 balls to verify that they all come out:: >>> balls = list(range(100)) >>> globe = ConcreteTombola(balls) >>> picks = [] >>> while globe.inspect(): ... picks.append(globe.pick()) >>> len(picks) == len(balls) True >>> set(picks) == set(balls) True Check that the order has changed and is not simply reversed:: >>> picks != balls True >>> picks[::-1] != balls True Note: the previous 2 tests have a *very* small chance of failing even if the implementation is OK. The probability of the 100 balls coming out, by chance, in the order they were inspect is 1/100!, or approximately 1.07e-158. It's much easier to win the Lotto or to become a billionaire working as a programmer. THE END
我们对 Tombola 抽象基类的分析到此结束。下一节说明 Python 如何使用抽象基类的 register 函数。
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论