python中的執行緒池
執行緒池的使用
執行緒池的基類是 concurrent.futures 模組中的 Executor,Executor 提供了兩個子類,即 ThreadPoolExecutor 和 ProcessPoolExecutor,其中 ThreadPoolExecutor 用於建立執行緒池,而 ProcessPoolExecutor 用於建立程式池。如果使用執行緒池/程式池來管理併發程式設計,那麼只要將相應的 task 函式提交給執行緒池/程式池,剩下的事情就由執行緒池/程式池來搞定。
Exectuor 提供瞭如下常用方法:
submit(fn, *args, **kwargs):將 fn 函式提交給執行緒池。*args 代表傳給 fn 函式的引數,*kwargs 代表以關鍵字引數的形式為 fn 函式傳入引數。
map(func, *iterables, timeout=None, chunksize=1):該函式類似於全域性函式 map(func, *iterables),只是該函式將會啟動多個執行緒,以非同步方式立即對 iterables 執行 map 處理。
shutdown(wait=True):關閉執行緒池。
程式將 task 函式提交(submit)給執行緒池後,submit 方法會返回一個 Future 物件,Future 類主要用於獲取執行緒任務函式的返回值。由於執行緒任務會在新執行緒中以非同步方式執行,因此,執行緒執行的函式相當於一個“將來完成”的任務,所以 Python 使用 Future 來代表。
實際上,在 的多執行緒程式設計中同樣有 Future,此處的 Future 與 Java 的 Future 大同小異。
Future 提供瞭如下方法:
cancel():取消該 Future 代表的執行緒任務。如果該任務正在執行,不可取消,則該方法返回 False;否則,程式會取消該任務,並返回 True。
cancelled():返回 Future 代表的執行緒任務是否被成功取消。
running():如果該 Future 代表的執行緒任務正在執行、不可被取消,該方法返回 True。
done():如果該 Funture 代表的執行緒任務被成功取消或執行完成,則該方法返回 True。
result(timeout=None):獲取該 Future 代表的執行緒任務最後返回的結果。如果 Future 代表的執行緒任務還未完成,該方法將會阻塞當前執行緒,其中 timeout 引數指定最多阻塞多少秒。
exception(timeout=None):獲取該 Future 代表的執行緒任務所引發的異常。如果該任務成功完成,沒有異常,則該方法返回 None。
add_done_callback(fn):為該 Future 代表的執行緒任務註冊一個“回撥函式”,當該任務成功完成時,程式會自動觸發該 fn 函式。
在用完一個執行緒池後,應該呼叫該執行緒池的 shutdown() 方法,該方法將啟動執行緒池的關閉序列。呼叫 shutdown() 方法後的執行緒池不再接收新任務,但會將以前所有的已提交任務執行完成。當執行緒池中的所有任務都執行完成後,該執行緒池中的所有執行緒都會死亡。
使用執行緒池來執行執行緒任務的步驟如下:
呼叫 ThreadPoolExecutor 類的構造器建立一個執行緒池。
定義一個普通函式作為執行緒任務。
呼叫 ThreadPoolExecutor 物件的 submit() 方法來提交執行緒任務。
當不想提交任何任務時,呼叫 ThreadPoolExecutor 物件的 shutdown() 方法來關閉執行緒池。
下面程式示範瞭如何使用執行緒池來執行執行緒任務:
from concurrent.futures import ThreadPoolExecutor import threading import time # 定義一個準備作為執行緒任務的函式 def action(max): my_sum = 0 for i in range(max): print(threading.current_thread().name + ' ' + str(i)) my_sum += i return my_sum # 建立一個包含2條執行緒的執行緒池 pool = ThreadPoolExecutor(max_workers=2) # 向執行緒池提交一個task, 50會作為action()函式的引數 future1 = pool.submit(action, 50) # 向執行緒池再提交一個task, 100會作為action()函式的引數 future2 = pool.submit(action, 100) # 判斷future1代表的任務是否結束 print(future1.done()) time.sleep(3) # 判斷future2代表的任務是否結束 print(future2.done()) # 檢視future1代表的任務返回的結果 print(future1.result()) # 檢視future2代表的任務返回的結果 print(future2.result()) # 關閉執行緒池 pool.shutdown()
當程式把 action() 函式提交給執行緒池時,submit() 方法會返回該任務所對應的 Future 物件,程式立即判斷 futurel 的 done() 方法,該方法將會返回 False(表明此時該任務還未完成)。接下來主程式暫停 3 秒,然後判斷 future2 的 done() 方法,如果此時該任務已經完成,那麼該方法將會返回 True。
程式最後透過 Future 的 result() 方法來獲取兩個非同步任務返回的結果。
讀者可以自己執行此程式碼檢視執行結果,這裡不再演示。
當程式使用 Future 的 result() 方法來獲取結果時,該方法會阻塞當前執行緒,如果沒有指定 timeout 引數,當前執行緒將一直處於阻塞狀態,直到 Future 代表的任務返回。獲取執行結果
前面程式呼叫了 Future 的 result() 方法來獲取執行緒任務的運回值,但該方法會阻塞當前主執行緒,只有等到錢程任務完成後,result() 方法的阻塞才會被解除。如果程式不希望直接呼叫 result() 方法阻塞執行緒,則可透過 Future 的 add_done_callback() 方法來新增回撥函式,該回撥函式形如 fn(future)。當執行緒任務完成後,程式會自動觸發該回撥函式,並將對應的 Future 物件作為引數傳給該回撥函式。
下面程式使用 add_done_callback() 方法來獲取執行緒任務的返回值:
from concurrent.futures import ThreadPoolExecutor import threading import time # 定義一個準備作為執行緒任務的函式 def action(max): my_sum = 0 for i in range(max): print(threading.current_thread().name + ' ' + str(i)) my_sum += i return my_sum # 建立一個包含2條執行緒的執行緒池 with ThreadPoolExecutor(max_workers=2) as pool: # 向執行緒池提交一個task, 50會作為action()函式的引數 future1 = pool.submit(action, 50) # 向執行緒池再提交一個task, 100會作為action()函式的引數 future2 = pool.submit(action, 100) def get_result(future): print(future.result()) # 為future1新增執行緒完成的回撥函式 future1.add_done_callback(get_result) # 為future2新增執行緒完成的回撥函式 future2.add_done_callback(get_result) print('--------------')
主程式的最後一行程式碼列印了一條橫線。由於程式並未直接呼叫 future1、future2 的 result() 方法,因此主執行緒不會被阻塞,可以立即看到輸出主執行緒列印出的橫線。接下來將會看到兩個新執行緒併發執行,當執行緒任務執行完成後,get_result() 函式被觸發,輸出執行緒任務的返回值。
另外,由於執行緒池實現了上下文管理協議(Context Manage Protocol),因此,程式可以使用 with 語句來管理執行緒池,這樣即可避免手動關閉執行緒池,如上面的程式所示。
此外,Exectuor 還提供了一個
map(func, *iterables, timeout=None, chunksize=1)
方法,該方法的功能類似於全域性函式 map(),區別在於執行緒池的 map() 方法會為 iterables 的每個元素啟動一個執行緒,以併發方式來執行 func 函式。這種方式相當於啟動 len(iterables) 個執行緒,井收集每個執行緒的執行結果。例如,如下程式使用 Executor 的 map() 方法來啟動執行緒,並收集執行緒任務的返回值:
from concurrent.futures import ThreadPoolExecutor import threading import time # 定義一個準備作為執行緒任務的函式 def action(max): my_sum = 0 for i in range(max): print(threading.current_thread().name + ' ' + str(i)) my_sum += i return my_sum # 建立一個包含4條執行緒的執行緒池 with ThreadPoolExecutor(max_workers=4) as pool: # 使用執行緒執行map計算 # 後面元組有3個元素,因此程式啟動3條執行緒來執行action函式 results = pool.map(action, (50, 100, 150)) print('--------------') for r in results: print(r)
上面程式使用 map() 方法來啟動 3 個執行緒(該程式的執行緒池包含 4 個執行緒,如果繼續使用只包含兩個執行緒的執行緒池,此時將有一個任務處於等待狀態,必須等其中一個任務完成,執行緒空閒出來才會獲得執行的機會),map() 方法的返回值將會收集每個執行緒任務的返回結果。
執行上面程式,同樣可以看到 3 個執行緒併發執行的結果,最後透過 results 可以看到 3 個執行緒任務的返回結果。
透過上面程式可以看出,使用 map() 方法來啟動執行緒,並收集執行緒的執行結果,不僅具有程式碼簡單的優點,而且雖然程式會以併發方式來執行 action() 函式,但最後收集的 action() 函式的執行結果,依然與傳入引數的結果保持一致。也就是說,上面 results 的第一個元素是 action(50) 的結果,第二個元素是 action(100) 的結果,第三個元素是 action(150) 的結果。
更多python相關文章,請關注。
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/2471/viewspace-2833430/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- Python的執行緒池Python執行緒
- Python執行緒池與程式池Python執行緒
- Python 執行緒池使用Python執行緒
- python執行緒池的實現Python執行緒
- Android中的執行緒池Android執行緒
- 詳解執行緒池的作用及Java中如何使用執行緒池執行緒Java
- Java執行緒池二:執行緒池原理Java執行緒
- java中執行緒池的生命週期與執行緒中斷Java執行緒
- 執行緒和執行緒池執行緒
- 執行緒 執行緒池 Task執行緒
- 多執行緒【執行緒池】執行緒
- java--執行緒池--建立執行緒池的幾種方式與執行緒池操作詳解Java執行緒
- 淺析Java中的執行緒池Java執行緒
- 執行緒池建立執行緒的過程執行緒
- Java多執行緒-執行緒池的使用Java執行緒
- Java中命名執行器服務執行緒和執行緒池Java執行緒
- 執行緒池執行緒
- SpringBoot執行緒池和Java執行緒池的實現原理Spring Boot執行緒Java
- 執行緒與執行緒池的那些事之執行緒池篇(萬字長文)執行緒
- 執行緒池以及四種常見執行緒池執行緒
- java執行緒池趣味事:這不是執行緒池Java執行緒
- Qt中的多執行緒與執行緒池淺析+例項QT執行緒
- 執行緒池的使用執行緒
- Python執行緒池ThreadPoolExecutor原始碼分析Python執行緒thread原始碼
- python實現自定義執行緒池Python執行緒
- springboot中如何使用執行緒池Spring Boot執行緒
- Python執行緒池 ThreadPoolExecutor 的用法及實戰Python執行緒thread
- 二. 執行緒管理之執行緒池執行緒
- Android多執行緒之執行緒池Android執行緒
- kuangshenshuo-多執行緒-執行緒池執行緒
- 多執行緒之手撕執行緒池執行緒
- java多執行緒9:執行緒池Java執行緒
- 聊聊面試中的 Java 執行緒池面試Java執行緒
- 執行緒池管理(1)-為什麼需要執行緒池執行緒
- Java併發 之 執行緒池系列 (1) 讓多執行緒不再坑爹的執行緒池Java執行緒
- 執行緒池的建立和使用,執行緒池原始碼初探(篇一)執行緒原始碼
- Android執行緒池Android執行緒
- java 執行緒池Java執行緒