python processpoolexecutor_python ThreadPoolExecutor线程池和ProcessPoolExecutor进程池

前言

Python标准库为我们提供了threading和multiprocessing模块编写相应的多线程/多进程代码,但是当项目达到一定的规模,频繁创建/销毁进程或者线程是非常消耗资源的,这个时候我们就要编写自己的线程池/进程池,以空间换时间。但从Python3.2开始,标准库为我们提供了concurrent.futures模块,它提供了ThreadPoolExecutor和ProcessPoolExecutor两个类,实现了对threading和multiprocessing的进一步抽象,对编写线程池/进程池提供了直接的支持。

相比 threading 等模块,该模块通过 submit 返回的是一个 future 对象,它是一个未来可期的对象,通过它可以获悉线程的状态主线程(或进程)中可以获取某一个线程(进程)执行的状态或者某一个任务执行的状态及返回值:

主线程可以获取某一个线程(或者任务的)的状态,以及返回值。

当一个线程完成的时候,主线程能够立即知道。

让多线程和多进程的编码接口一致。

线程池的使用

from concurrent.futures import ThreadPoolExecutor

import time

def task(i):

time.sleep(i)

print(f'task{i} completed')

pass

thread_executor = ThreadPoolExecutor(max_workers=7) # 创建一个最大容纳数量为7的线程池

start = time.time()

task1 = thread_executor.submit(task, 0)

task1 = thread_executor.submit(task, 1)

task1 = thread_executor.submit(task, 2)

task1 = thread_executor.submit(task, 3)

task1 = thread_executor.submit(task, 4)

# 打印该任务是否执行完毕

print(task1.done())

# 只有未被提交的到线程池(在等待提交的队列中)的任务才能够取消

print(task3.cancel())

time.sleep(4) # 休眠4秒钟之后,线程池中的任务全部执行完毕,可以打印状态

print(task1.done())

print(task1.result()) # 该任务的return 返回值 该方法是阻塞的。

thread_executor.shutdown(wait=True)

end = time.time()

print(f'Time consuming {end - start}')

执行结果

task0 completed

False

False

task1 completed

task2 completed

task3 completed

True

None

task4 completed

Time consuming 4.002375364303589

ThreadPoolExecutor构造实例的时候,传入max_workers参数来设置线程池中最多能同时运行的线程数目。

使用submit函数来提交线程需要执行的任务(函数名和参数)到线程池中,并返回该任务的句柄(类似于文件、画图),注意submit()不是阻塞的,而是立即返回。

通过submit函数返回的任务句柄,能够使用done()方法判断该任务是否结束。上面的例子可以看出,由于任务有2s的延时,在task1提交后立刻判断,task1还未完成,而在延时4s之后判断,task1就完成了。

使用cancel()方法可以取消提交的任务,如果任务已经在线程池中运行了,就取消不了。这个例子中,线程池的大小设置为2,任务已经在运行了,所以取消失败。如果改变线程池的大小为1,那么先提交的是task1,task2还在排队等候,这是时候就可以成功取消。

使用result()方法可以获取任务的返回值。查看内部代码,发现这个方法是阻塞的。

as_completed

上面虽然提供了判断任务是否结束的方法,但是不能在主线程中一直判断啊。有时候我们是得知某个任务结束了,就去获取结果,而不是一直判断每个任务有没有结束。这是就可以使用as_completed方法一次取出所有任务的结果。

as_completed()方法是一个生成器,在没有任务完成的时候,会阻塞,在有某个任务完成的时候,会yield这个任务,就能执行for循环下面的语句,然后继续阻塞住,循环到所有的任务结束。从结果也可以看出,先完成的任务会先通知主线程。

map

除了上面的as_completed方法,还可以使用executor.map方法,但是有一点不同。

from concurrent.futures import ThreadPoolExecutor

import time

def task(i):

time.sleep(i)

print(f'task{i} completed')

pass

thread_executor = ThreadPoolExecutor(max_workers=7) # 创建一个最大容纳数量为7的线程池

start = time.time()

params = [i for i in range(5)]

thread_executor.map(task, params)

thread_executor.shutdown(wait=True)

end = time.time()

print(f'Time consuming {end - start}')

执行结果

task0 completed

task1 completed

task2 completed

task3 completed

task4 completed

Time consuming 4.002383232116699

ProcessPoolExecutor使用

ProcessPoolExecutor在使用上和ThreadPoolExecutor大致是一样的,它们在futures中的方法也是相同的,但是对于map()方法ProcessPoolExecutor会多一个参数chunksize(ThreadPoolExecutor中这个参数没有任何作用),chunksize将迭代对象切成块,将其作为分开的任务提交给pool,对于很大的iterables,设置较大chunksize可以提高性能。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Python中的进程线程池是用来进行并发处理任务的工具。进程是在一个子中放入固定数量的进程,当有任务到达时,就从子中拿一个进程来处理任务,处理完毕后再放回子中等待任务。进程的数量是固定的,因此同一时间最多有固定数量的进程在运行。这样可以避免增加操作系统的调度难度,并节省开闭进程的时间,实现并发效果。 在Python中,可以使用`concurrent.futures`模块中的`ProcessPoolExecutor`类来创建进程。通过实例化`ProcessPoolExecutor`类,可以获得一个指定大小的进程。例如,可以使用以下代码实例化一个进程并指定大小为整数`n`: ``` from concurrent.futures import ProcessPoolExecutor pool = ProcessPoolExecutor(n) ``` 其中`n`是进程的大小,即进程的数量。 线程池进程的原理类似,不同之处在于线程池是在一个子中放入固定数量的线程来处理任务。线程池的使用方式也类似,可以使用`concurrent.futures`模块中的`ThreadPoolExecutor`类来创建线程池,并指定大小。线程池的使用和进程类似,都是从子中获取线程或进程来处理任务。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [Python--进程线程池](https://blog.csdn.net/weixin_43988680/article/details/124284555)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值