FluentPy Note(2)
可迭代对象、迭代器和生成器
迭代器模式:惰性获取数据项的方式。
所有生成器都是迭代器,因为生成器完全实现了迭代器接口。在 Python社区中,大多数时候都把迭代器和生成器视作同一概念。
生成器应用广泛,即使是内置的range()也返回类似生成器对象,如果一定要返回列表,必须明确指明(例如,list(range(100)))。
序列可以迭代的原因:iter函数。
内置的iter函数有以下作用。
- 检查对象是否实现了
__iter__
方法,如果实现了即调用,获取一个迭代器。 - 如果没有实现
__iter__
方法,但实现了__getitem__
方法。Python会创建一个迭代器,尝试按顺序(从索引0开始)获取元素。 - 失败,抛出TypeError异常。
任何 Python 序列都可迭代的原因是,它们都实现了__getitem__
方法。其实,标准的序列也都实现了 __iter__
方法,因此你也应该这么做。之所以对 __getitem__
方法做特殊处理,是为了向后兼容,而未来可能不会再这么做(不过,写作本书时还未弃用)。
检查对象 x 能否迭代,最准确的方法是:调用 iter(x) 函数,如果不可迭代,再处理 TypeError 异常。
实现标准的可迭代协议
1 | class Sentence: |
Python函数中有yield生成器函数,返回生成器。
生成器表达式可以理解为列表推导的惰性版本:不会迫切地构建列表,而是返回一个生成器,按需惰性生成元素。
标准库中的生成器函数
按用途可分为以下几类:
- 过滤
- 映射
- 合并
1 | def vowel(c): |
yield from 语句。这个语句的作用就是把不同的生成器结合在一起使用。
上下文管理器和 else 块
with open 创建文件不创建路径。
仅当 try 块中没有异常抛出时才运行 else 块。
1 | try: |
with语句的目的是简化try/finally模式。
与函数和模块不同,with 块没有定义新的作用域。
多线程与协程
标准库中所有执行阻塞型 I/O 操作的函数,在等待操作系统返回结果时都会释放 GIL。这意味着在 Python 语言这个层次上可以使用多线程,而 I/O 密集型 Python 程序能从中受益:一个 Python 线程等待网络响应时,阻塞型 I/O 函数会释放 GIL,再运行一个线程。
Concurrency is when two or more tasks can start, run, and complete in overlapping time periods. It doesn’t necessarily mean they’ll ever both be running at the same instant. For example, multitaskingon a single-core machine.
Parallelism is when tasks literally run at the same time, e.g., on a multicore processor.
异步库依赖于低层线程(直至内核级线程),但是这些库的用户无需创建线程,也无需知道用到了基础设施中的低层线程。在应用中,我们只需确保没有阻塞的代码,事件循环会在背后处理并发。异步系统能避免用户级线程的开销,这是它能比多线程系统管理更多并发连接的主要原因。
并发:充分运用CPU资源,主要是针对线程。
Coroutines
1 | async def say_after(delay, what): |
asyncio.run(coro,*,debug=False)
当前线程存在一个事件循环时,该函数不能调用。
该函数调用时新建一个事件循环并在结束时关闭循环。它应该被用作异步程序的主入口并且理想情况下,应该只调用一次。
Task Object
class asyncio.Task(coro,*,loop=None)
Task用于在事件循环中运行协程。当一个协程等待一个Future时,Task中断协程的执行并等待Future完成。当Future完成,协程恢复。
事件循环使用竞争调度:一个事件循环同一时间只运行一个任务。然而,当一任务等待Future的完成时,事件循环运行其他任务等。
asyncio.create_task(coro)
把协程包装成Task并调度其执行。Task在事件循环中执行,这个循环通过get_running_loop()获得,若当前线程没有事件循环,抛出RuntimeError。
1 | async def say_after(delay, what): |
Awaitables
There are three main types of awaitable objects: coroutines, Tasks, and Futures.
Tasks are used to schedule coroutines concurrently.
A Future is a special low-level awaitable object that represents an eventual result of an asynchronous operation.
Normally there is no need to create Future objects at the application level code.
coroutine asyncio.sleep(delay, result=None, *, loop=None)
sleep()
always suspends the current task, allowing other tasks to run.
Running Tasks Concurrently
awaitable asyncio.gather(*aws, loop=None, return_exceptions=False)
并发执行aws序列中的awaitable对象。
如果aws中有协程,自动当作Task调度。
If all awaitables are completed successfully, the result is an aggregate list of returned values. The order of result values corresponds to the order of awaitables in aws.
If return_exceptions is False (default), the first raised exception is immediately propagated to the task that awaits on gather(). Other awaitables in the aws sequence won’t be cancelled and will continue to run.
If return_exceptions is True, exceptions are treated the same as successful results, and aggregated in the result list.
If gather() is cancelled, all submitted awaitables (that have not completed yet) are also cancelled.
If any Task or Future from the aws sequence is cancelled, it is treated as if it raised CancelledError – the gather() call is not cancelled in this case. This is to prevent the cancellation of one submitted Task/Future to cause other Tasks/Futures to be cancelled.
元编程
使用点号访问属性时,Python解释器会调用特殊方法(__getattr__
和__setattr__
)计算属性。
特性:在不改变类接口的前提下,使用存取方法(即读值方法和设值方法)修改数据属性。
使用特性验证属性
@property
1 | class LineItem: |
被装饰的读值方法有个 .setter 属性,这个属性也是装饰器;这个装饰器把读值方法和设值方法绑定在一起。
虽然property经常被当作装饰器使用,但它其实是一个类。构造方法如下:
property(fget=None, fset=None, fdel=None, doc=None)
老版Python这样用:weight = property(get_weight, set_weight)
特性会覆盖实例属性
特性都是类属性,但特性管理的其实是实例属性的存取。
实例属性遮盖类的数据属性,但实例属性不会遮盖类特性。
1 | class Class: |
1 | # 获取特性对象本身,不会运行特性读值方法 Class.prop |
定义特性工厂函数
1 | def quantity(storage_name): |
影响属性处理方式的特殊属性
__class__
对象所属类的引用,Python的__getattr__
只在类中寻找,而不在实例中寻找。__getattr__
只处理不存在的属性名。__dict__
存储对象或类的可写属性。
特殊方法
__delattr__(self, name)
只要使用 del 语句删除属性,就会调用这个方法。__getattr__(self, name)
仅当获取指定的属性失败,搜索过 obj、Class 和超类之后调用。表达式 obj.no_such_attr、getattr(obj, ‘no_such_attr’) 和hasattr(obj, ‘no_such_attr’) 可能会触发Class.__getattr__(obj, 'no_such_attr')
方法,但是,仅当在
obj、Class 和超类中找不到指定的属性时才会触发。__getattribute__(self, name)
点号与 getattr 和 hasattr 内置函数会触发这个方法。尝试获取指定的属性时总会调用这个方法,不过,寻找的属性是特殊属性或特殊方法时除外。调用__getattribute__
方法且抛出 AttributeError异常时,才会调用__getattr__
方法。__setattr__(self, name, value)
点号和 setattr 内置函数会触发这个方法。
属性描述符
描述符是对多个属性运行相同存取逻辑的一种方法。
描述符是实现了特定协议的类,这个协议包括__get__
、__set__
和__delete__
。其实,我们在真实的代码中见到的大多数描述符只实现了 __get__
和__set__
方法,还有很多只实现了其中的一个。
property 类实现了完整的描述符协议。
特性工厂函数借助函数式编程模式避免重复编写读值方法和设值方法。解决这种问题的面向对象的方法是描述符类。
1 | class Quantity: #描述符类基于协议实现 |
类元编程
类元编程是指在运行时创建或定制类的技术。
类装饰器也是函数,不过能够审查、修改,甚至把被装饰的类替换成其他类。
导入时和运行时区别——是有效使用Python元编程的重要基础。
类装饰器
1 | class AutoStorage: |
1 | import model_v5 as model |
类装饰器以较简单的方式做到以前元类去做的事情——创建类时定制类。
类装饰器有个重大缺点:只对直接依附的类有效。这意味着,被装饰的类的子类可能继承也可能不继承装饰器所做的改动,具体情况视改动的方式而定。
元类
用于构建类的类。Python中,类也是对象,因此类必然是某个类的实例。默认情况下,Python类是type类的实例。也就是说,type 是大多数内置的类和用户定义的类的元类。为了避免无限回溯,type 是其自身的实例。