concurrent.futures
--- 启动并行任务¶
Added in version 3.2.
源码: Lib/concurrent/futures/thread.py 和 Lib/concurrent/futures/process.py
concurrent.futures
模块提供异步执行可调用对象高层接口。
The asynchronous execution can be performed with threads, using
ThreadPoolExecutor
or InterpreterPoolExecutor
,
or separate processes, using ProcessPoolExecutor
.
Each implements the same interface, which is defined
by the abstract Executor
class.
Availability: not WASI.
此模块在 WebAssembly 平台上无效或不可用。 请参阅 WebAssembly 平台 了解详情。
Executor 对象¶
- class concurrent.futures.Executor¶
抽象类提供异步执行调用方法。要通过它的子类调用,而不是直接调用。
- submit(fn, /, *args, **kwargs)¶
调度可调用对象 fn,以
fn(*args, **kwargs)
方式执行并返回一个代表该可调用对象的执行的Future
对象。with ThreadPoolExecutor(max_workers=1) as executor: future = executor.submit(pow, 323, 1235) print(future.result())
- map(fn, *iterables, timeout=None, chunksize=1)¶
类似于
map(fn, *iterables)
但有以下差异:iterables 是立即执行而不是延迟执行的;
fn 是异步执行的并且可以并发对 fn 的多个调用。
如果
__next__()
被调用且从对Executor.map()
原始调用 timeout 秒之后其结果还不可用则已返回的迭代器将引发TimeoutError
。 timeout 可以是整数或浮点数。 如果 timeout 未指定或为None
,则不限制等待时间。如果 fn 调用引发了异常,那么当从迭代器获取其值时该异常将被引发。
When using
ProcessPoolExecutor
, this method chops iterables into a number of chunks which it submits to the pool as separate tasks. The (approximate) size of these chunks can be specified by setting chunksize to a positive integer. For very long iterables, using a large value for chunksize can significantly improve performance compared to the default size of 1. WithThreadPoolExecutor
andInterpreterPoolExecutor
, chunksize has no effect.在 3.5 版本发生变更: 加入 chunksize 参数。
- shutdown(wait=True, *, cancel_futures=False)¶
当待执行的 future 对象完成执行后向执行者发送信号,它就会释放正在使用的任何资源。 在关闭后调用
Executor.submit()
和Executor.map()
将会引发RuntimeError
。如果 wait 为
True
则此方法只有在所有待执行的 future 对象完成执行且释放已分配的资源后才会返回。 如果 wait 为False
,方法立即返回,所有待执行的 future 对象完成执行后会释放已分配的资源。 不管 wait 的值是什么,整个 Python 程序将等到所有待执行的 future 对象完成执行后才退出。如果 cancel_futures 为
True
,此方法将取消所有执行器还未开始运行的挂起的 Future。无论 cancel_futures 的值是什么,任何已完成或正在运行的 Future 都不会被取消。如果 cancel_futures 和 wait 均为
True
,则执行器已开始运行的所有 Future 将在此方法返回之前完成。 其余的 Future 会被取消。如果使用
with
语句,你就可以避免显式调用这个方法,它将会停止Executor
(就好像Executor.shutdown()
调用时 wait 设为True
一样等待):import shutil with ThreadPoolExecutor(max_workers=4) as e: e.submit(shutil.copy, 'src1.txt', 'dest1.txt') e.submit(shutil.copy, 'src2.txt', 'dest2.txt') e.submit(shutil.copy, 'src3.txt', 'dest3.txt') e.submit(shutil.copy, 'src4.txt', 'dest4.txt')
在 3.9 版本发生变更: 增加了 cancel_futures。
ThreadPoolExecutor¶
ThreadPoolExecutor
是 Executor
的子类,它使用线程池来异步执行调用。
当可调用对象已关联了一个 Future
然后在等待另一个 Future
的结果时就会导致死锁情况。例如:
import time
def wait_on_b():
time.sleep(5)
print(b.result()) # b 永远不会结束因为它在等待 a。
return 5
def wait_on_a():
time.sleep(5)
print(a.result()) # a 永远不会结束因为它在等待 b。
return 6
executor = ThreadPoolExecutor(max_workers=2)
a = executor.submit(wait_on_b)
b = executor.submit(wait_on_a)
与:
def wait_on_future():
f = executor.submit(pow, 5, 2)
# 这将永远不会完成因为只有一个工作线程
# 并且它正在执行此函数。
print(f.result())
executor = ThreadPoolExecutor(max_workers=1)
executor.submit(wait_on_future)
- class concurrent.futures.ThreadPoolExecutor(max_workers=None, thread_name_prefix='', initializer=None, initargs=())¶
Executor
子类使用最多 max_workers 个线程的线程池来异步执行调用。所有排入
ThreadPoolExecutor
的队列的线程将在解释器退出之前被合并。 请注意执行此操作的 exit 处理器会在任何使用atexit
添加的 exit处理器 之前 被执行。 这意味着主线程中的异常必须被捕获和处理以便向线程发出信号使其能够优雅地退出。 由于这个原理,建议不要将ThreadPoolExecutor
用于长期运行的任务。initializer 是在每个工作者线程开始处调用的一个可选可调用对象。 initargs 是传递给初始化器的元组参数。任何向池提交更多工作的尝试, initializer 都将引发一个异常,当前所有等待的工作都会引发一个
BrokenThreadPool
。在 3.5 版本发生变更: 如果 max_workers 为
None
或没有指定,将默认为机器处理器的个数,假如ThreadPoolExecutor
侧重于I/O操作而不是CPU运算,那么可以乘以5
,同时工作线程的数量可以比ProcessPoolExecutor
的数量高。在 3.6 版本发生变更: 增加了 thread_name_prefix 形参来允许用户控制由线程池创建的
threading.Thread
工作线程名称以方便调试。在 3.7 版本发生变更: 加入 initializer 和*initargs* 参数。
在 3.8 版本发生变更: max_workers 的默认值已改为
min(32, os.cpu_count() + 4)
。 这个默认值会保留至少 5 个工作线程用于 I/O 密集型任务。 对于那些释放了 GIL 的 CPU 密集型任务,它最多会使用 32 个 CPU 核心。这样能够避免在多核机器上不知不觉地使用大量资源。现在 ThreadPoolExecutor 在启动 max_workers 个工作线程之前也会重用空闲的工作线程。
在 3.13 版本发生变更: max_workers 的默认值已改为
min(32, (os.process_cpu_count() or 1) + 4)
。
ThreadPoolExecutor 例子¶
import concurrent.futures
import urllib.request
URLS = ['http://www.foxnews.com/',
'http://www.cnn.com/',
'http://europe.wsj.com/',
'http://www.bbc.co.uk/',
'http://nonexistent-subdomain.python.org/']
# Retrieve a single page and report the URL and contents
def load_url(url, timeout):
with urllib.request.urlopen(url, timeout=timeout) as conn:
return conn.read()
# We can use a with statement to ensure threads are cleaned up promptly
with concurrent.futures.ThreadPoolExecutor(max_workers=5) as executor:
# Start the load operations and mark each future with its URL
future_to_url = {executor.submit(load_url, url, 60): url for url in URLS}
for future in concurrent.futures.as_completed(future_to_url):
url = future_to_url[future]
try:
data = future.result()
except Exception as exc:
print('%r generated an exception: %s' % (url, exc))
else:
print('%r page is %d bytes' % (url, len(data)))
InterpreterPoolExecutor¶
The InterpreterPoolExecutor
class uses a pool of interpreters
to execute calls asynchronously. It is a ThreadPoolExecutor
subclass, which means each worker is running in its own thread.
The difference here is that each worker has its own interpreter,
and runs each task using that interpreter.
The biggest benefit to using interpreters instead of only threads is true multi-core parallelism. Each interpreter has its own Global Interpreter Lock, so code running in one interpreter can run on one CPU core, while code in another interpreter runs unblocked on a different core.
The tradeoff is that writing concurrent code for use with multiple interpreters can take extra effort. However, this is because it forces you to be deliberate about how and when interpreters interact, and to be explicit about what data is shared between interpreters. This results in several benefits that help balance the extra effort, including true multi-core parallelism, For example, code written this way can make it easier to reason about concurrency. Another major benefit is that you don't have to deal with several of the big pain points of using threads, like nrace conditions.
Each worker's interpreter is isolated from all the other interpreters.
"Isolated" means each interpreter has its own runtime state and
operates completely independently. For example, if you redirect
sys.stdout
in one interpreter, it will not be automatically
redirected any other interpreter. If you import a module in one
interpreter, it is not automatically imported in any other. You
would need to import the module separately in interpreter where
you need it. In fact, each module imported in an interpreter is
a completely separate object from the same module in a different
interpreter, including sys
, builtins
,
and even __main__
.
Isolation means a mutable object, or other data, cannot be used by more than one interpreter at the same time. That effectively means interpreters cannot actually share such objects or data. Instead, each interpreter must have its own copy, and you will have to synchronize any changes between the copies manually. Immutable objects and data, like the builtin singletons, strings, and tuples of immutable objects, don't have these limitations.
Communicating and synchronizing between interpreters is most effectively
done using dedicated tools, like those proposed in PEP 734. One less
efficient alternative is to serialize with pickle
and then send
the bytes over a shared socket
or
pipe
.
- class concurrent.futures.InterpreterPoolExecutor(max_workers=None, thread_name_prefix='', initializer=None, initargs=(), shared=None)¶
A
ThreadPoolExecutor
subclass that executes calls asynchronously using a pool of at most max_workers threads. Each thread runs tasks in its own interpreter. The worker interpreters are isolated from each other, which means each has its own runtime state and that they can't share any mutable objects or other data. Each interpreter has its own Global Interpreter Lock, which means code run with this executor has true multi-core parallelism.The optional initializer and initargs arguments have the same meaning as for
ThreadPoolExecutor
: the initializer is run when each worker is created, though in this case it is run.in the worker's interpreter. The executor serializes the initializer and initargs usingpickle
when sending them to the worker's interpreter.备注
Functions defined in the
__main__
module cannot be pickled and thus cannot be used.备注
The executor may replace uncaught exceptions from initializer with
ExecutionFailed
.The optional shared argument is a
dict
of objects that all interpreters in the pool share. The shared items are added to each interpreter's__main__
module. Not all objects are shareable. Shareable objects include the builtin singletons,str
andbytes
, andmemoryview
. See PEP 734 for more info.Other caveats from parent
ThreadPoolExecutor
apply here.
submit()
and map()
work like normal,
except the worker serializes the callable and arguments using
pickle
when sending them to its interpreter. The worker
likewise serializes the return value when sending it back.
备注
Functions defined in the __main__
module cannot be pickled
and thus cannot be used.
When a worker's current task raises an uncaught exception, the worker
always tries to preserve the exception as-is. If that is successful
then it also sets the __cause__
to a corresponding
ExecutionFailed
instance, which contains a summary of the original exception.
In the uncommon case that the worker is not able to preserve the
original as-is then it directly preserves the corresponding
ExecutionFailed
instance instead.
ProcessPoolExecutor¶
ProcessPoolExecutor
类是 Executor
的子类,它使用进程池来异步地执行调用。 ProcessPoolExecutor
会使用 multiprocessing
模块,这允许它绕过 全局解释器锁 但也意味着只可以处理和返回可封存的对象。
__main__
模块必须可以被工作者子进程导入。这意味着 ProcessPoolExecutor
不可以工作在交互式解释器中。
从可调用对象中调用 Executor
或 Future
的方法提交给 ProcessPoolExecutor
会导致死锁。
- class concurrent.futures.ProcessPoolExecutor(max_workers=None, mp_context=None, initializer=None, initargs=(), max_tasks_per_child=None)¶
异步地执行调用的
Executor
子类使用最多 max_workers 个进程的进程池。 如果 max_workers 为None
或未给出,它将默认为os.process_cpu_count()
。 如果 max_workers 小于等于0
,则将引发ValueError
。 在 Windows 上,max_workers 必须小于等于61
。 如果不是这样则将引发ValueError
。 如果 max_workers 为None
,则选择的默认值最多为61
,即使存在更多的处理器。 mp_context 可以是一个multiprocessing
上下文或是None
。 它将被用来启动工作进程。 如果 mp_context 为None
或未给出,则将使用默认的multiprocessing
上下文。 参见 上下文和启动方法。initializer 是一个可选的可调用对象,它会在每个工作进程启动时被调用;initargs 是传给 initializer 的参数元组。 如果 initializer 引发了异常,则所有当前在等待的任务以及任何向进程池提交更多任务的尝试都将引发
BrokenProcessPool
。max_tasks_per_child 是指定单个进程在其退出并替换为新工作进程之前可以执行的最大任务数量的可选参数。 在默认情况下 max_tasks_per_child 为
None
表示工作进程将存活与进程池一样长的时间。 当指定了最大数量时,则如果不存在 mp_context 形参则将默认使用 "spawn" 多进程启动方法。 此特性不能兼容 "fork" 启动方法。在 3.3 版本发生变更: 当某个工作进程突然终止时,现在将引发
BrokenProcessPool
。 在之前版本中,它的行为是未定义的但在执行器上的操作或它的 future 对象往往会被冻结或死锁。在 3.7 版本发生变更: 添加 mp_context 参数允许用户控制由进程池创建给工作者进程的开始方法 。
加入 initializer 和*initargs* 参数。
在 3.11 版本发生变更: 增加了 max_tasks_per_child 参数以允许用户控制进程池中工作进程的生命期。
在 3.12 版本发生变更: 在 POSIX 系统上,如果你的应用程序有多个线程而
multiprocessing
上下文使用了"fork"
启动方法:内部调用的os.fork()
函数来生成工作进程可能会引发DeprecationWarning
。 请传递配置为使用不同启动方法的 mp_context。 进一步的解释请参阅os.fork()
文档。在 3.13 版本发生变更: 在默认情况下 max_workers 将使用
os.process_cpu_count()
,而不是os.cpu_count()
。在 3.14 版本发生变更: The default process start method (see 上下文和启动方法) changed away from fork. If you require the fork start method for
ProcessPoolExecutor
you must explicitly passmp_context=multiprocessing.get_context("fork")
.
ProcessPoolExecutor 例子¶
import concurrent.futures
import math
PRIMES = [
112272535095293,
112582705942171,
112272535095293,
115280095190773,
115797848077099,
1099726899285419]
def is_prime(n):
if n < 2:
return False
if n == 2:
return True
if n % 2 == 0:
return False
sqrt_n = int(math.floor(math.sqrt(n)))
for i in range(3, sqrt_n + 1, 2):
if n % i == 0:
return False
return True
def main():
with concurrent.futures.ProcessPoolExecutor() as executor:
for number, prime in zip(PRIMES, executor.map(is_prime, PRIMES)):
print('%d is prime: %s' % (number, prime))
if __name__ == '__main__':
main()
Future 对象¶
Future
类将可调用对象封装为异步执行。Future
实例由 Executor.submit()
创建。
- class concurrent.futures.Future¶
将可调用对象封装为异步执行。
Future
实例由Executor.submit()
创建,除非测试,不应直接创建。- cancel()¶
尝试取消调用。 如果调用正在执行或已结束运行不能被取消则该方法将返回
False
,否则调用会被取消并且该方法将返回True
。
- cancelled()¶
如果调用成功取消返回
True
。
- running()¶
如果调用正在执行而且不能被取消那么返回
True
。
- done()¶
如果调用已被取消或正常结束那么返回
True
。
- result(timeout=None)¶
返回调用所返回的值。 如果调用尚未完成则此方法将等待至多 timeout 秒。 如果调用在 timeout 秒内仍未完成,则将引发
TimeoutError
。 timeout 可以为整数或浮点数。 如果 timeout 未指定或为None
,则不限制等待时间。如果 future 在完成前被取消则
CancelledError
将被触发。如果调用引发了一个异常,这个方法也会引发同样的异常。
- exception(timeout=None)¶
返回调用所引发的异常。 如果调用尚未完成则此方法将等待至多 timeout 秒。 如果调用在 timeout 秒内仍未完成,则将引发
TimeoutError
。 timeout 可以为整数或浮点数。 如果 timeout 未指定或为None
,则不限制等待时间。如果 future 在完成前被取消则
CancelledError
将被触发。如果调用正常完成那么返回
None
。
- add_done_callback(fn)¶
附加可调用 fn 到 future 对象。当 future 对象被取消或完成运行时,将会调用 fn,而这个 future 对象将作为它唯一的参数。
加入的可调用对象总被属于添加它们的进程中的线程按加入的顺序调用。如果可调用对象引发一个
Exception
子类,它会被记录下来并被忽略掉。如果可调用对象引发一个BaseException
子类,这个行为没有定义。如果 future 对象已经完成或已取消,fn 会被立即调用。
下面这些
Future
方法用于单元测试和Executor
实现。- set_running_or_notify_cancel()¶
这个方法只可以在执行关联
Future
工作之前由Executor
实现调用或由单测试调用。如果此方法返回
False
则Future
已被取消,即Future.cancel()
已被调用并返回True
。 任何等待Future
完成 (即通过as_completed()
或wait()
) 的线程将被唤醒。如果此方法返回
True
则Future
没有被取消并已被置为正在运行的状态,即对Future.running()
的调用将返回True
。这个方法只可以被调用一次并且不能在调用
Future.set_result()
或Future.set_exception()
之后再调用。
- set_result(result)¶
设置将
Future
关联工作的结果给 result 。这个方法只可以由
Executor
实现和单元测试使用。在 3.8 版本发生变更: 如果
Future
已经完成则此方法会引发concurrent.futures.InvalidStateError
。
模块函数¶
- concurrent.futures.wait(fs, timeout=None, return_when=ALL_COMPLETED)¶
等待由 fs 指定的
Future
实例(可能由不同的Executor
实例创建)完成。 重复传给 fs 的 future 会被移除并将只返回一次。 返回一个由集合组成的具名 2 元组。 第一个集合的名称为done
,包含在等待完成之前已完成的 future(包括正常结束或被取消的 future)。 第二个集合的名称为not_done
,包含未完成的 future(包括挂起的或正在运行的 future)。timeout 可以用来控制返回前最大的等待秒数。 timeout 可以为 int 或 float 类型。 如果 timeout 未指定或为
None
,则不限制等待时间。return_when 指定此函数应在何时返回。它必须为以下常数之一:
常量
描述
- concurrent.futures.FIRST_COMPLETED¶
函数将在任意可等待对象结束或取消时返回。
- concurrent.futures.FIRST_EXCEPTION¶
该函数将在任何 future 对象通过引发异常而结束时返回。 如果没有任何 future 对象引发引发那么它将等价于
ALL_COMPLETED
。- concurrent.futures.ALL_COMPLETED¶
函数将在所有可等待对象结束或取消时返回。
- concurrent.futures.as_completed(fs, timeout=None)¶
返回一个迭代器,每当 fs 所给出的
Future
实例(可能由不同的Executor
实例创建)完成时这个迭代器会产生新的 future(包括正常结束或被取消的 future 对象)。 任何由 fs 给出的重复的 future 对象将只被返回一次。 任何在as_completed()
被调用之前完成的 future 对象将优先被产生。 如果__next__()
被调用并且在最初调用as_completed()
之后的 timeout 秒内其结果仍不可用,这个迭代器将引发TimeoutError
。 timeout 可以为整数或浮点数。 如果 timeout 未指定或为None
,则不限制等待时间。
参见
- PEP 3148 -- future 对象 - 异步执行指令。
该提案描述了Python标准库中包含的这个特性。
Exception 类¶
- exception concurrent.futures.CancelledError¶
future 对象被取消时会触发。
- exception concurrent.futures.TimeoutError¶
TimeoutError
的一个已被弃用的别名,会在 future 操作超出了给定的时限时被引发。在 3.11 版本发生变更: 这个类是
TimeoutError
的别名。
- exception concurrent.futures.BrokenExecutor¶
当执行器被某些原因中断而且不能用来提交或执行新任务时就会被引发派生于
RuntimeError
的异常类。Added in version 3.7.
- exception concurrent.futures.InvalidStateError¶
当某个操作在一个当前状态所不允许的 future 上执行时将被引发。
Added in version 3.8.
- exception concurrent.futures.thread.BrokenThreadPool¶
派生自
BrokenExecutor
,这个异常类会在ThreadPoolExecutor
的某个工作线程初始化失败时被引发。Added in version 3.7.
- exception concurrent.futures.interpreter.BrokenInterpreterPool¶
Derived from
BrokenThreadPool
, this exception class is raised when one of the workers of aInterpreterPoolExecutor
has failed initializing.Added in version 3.14.0a1 (unreleased).
- exception concurrent.futures.interpreter.ExecutionFailed¶
Raised from
InterpreterPoolExecutor
when the given initializer fails or fromsubmit()
when there's an uncaught exception from the submitted task.Added in version 3.14.0a1 (unreleased).
- exception concurrent.futures.process.BrokenProcessPool¶
派生自
BrokenExecutor
(原为RuntimeError
),这个异常类会在ProcessPoolExecutor
的某个工作进程以不完整的方式终结(例如,从外部杀掉)时被引发。Added in version 3.3.