詳解Python獲取線程返回值的三種方式
提到線程,你的大腦應(yīng)該有這樣的印象:我們可以控制它何時(shí)開(kāi)始,卻無(wú)法控制它何時(shí)結(jié)束,那么如何獲取線程的返回值呢?今天就分享一下自己的一些做法。
方法一
使用全局變量的列表,來(lái)保存返回值
ret_values?=?[] def?thread_func(*args): ????... ????value?=?... ????ret_values.append(value)
選擇列表的一個(gè)原因是:列表的 append() 方法是線程安全的,CPython 中,GIL 防止對(duì)它們的并發(fā)訪問(wèn)。如果你使用自定義的數(shù)據(jù)結(jié)構(gòu),在并發(fā)修改數(shù)據(jù)的地方需要加線程鎖。
如果事先知道有多少個(gè)線程,可以定義一個(gè)固定長(zhǎng)度的列表,然后根據(jù)索引來(lái)存放返回值,比如:
from?threading?import?Thread
threads?=?[None]?*?10
results?=?[None]?*?10
def?foo(bar,?result,?index):
????result[index]?=?f"foo-{index}"
for?i?in?range(len(threads)):
????threads[i]?=?Thread(target=foo,?args=('world!',?results,?i))
????threads[i].start()
for?i?in?range(len(threads)):
????threads[i].join()
print?("?".join(results))
方法二
重寫(xiě) Thread 的 join 方法,返回線程函數(shù)的返回值
默認(rèn)的 thread.join() 方法只是等待線程函數(shù)結(jié)束,沒(méi)有返回值,我們可以在此處返回函數(shù)的運(yùn)行結(jié)果,代碼如下:
from?threading?import?Thread
def?foo(arg):
????return?arg
class?ThreadWithReturnValue(Thread):
????def?run(self):
????????if?self._target?is?not?None:
????????????self._return?=?self._target(*self._args,?**self._kwargs)
????def?join(self):
????????super().join()
????????return?self._return
twrv?=?ThreadWithReturnValue(target=foo,?args=("hello?world",))
twrv.start()
print(twrv.join())?#?此處會(huì)打印 hello world。
這樣當(dāng)我們調(diào)用 thread.join() 等待線程結(jié)束的時(shí)候,也就得到了線程的返回值。
方法三
使用標(biāo)準(zhǔn)庫(kù) concurrent.futures
我覺(jué)得前兩種方式實(shí)在太低級(jí)了,Python 的標(biāo)準(zhǔn)庫(kù) concurrent.futures 提供更高級(jí)的線程操作,可以直接獲取線程的返回值,相當(dāng)優(yōu)雅,代碼如下:
import?concurrent.futures
def?foo(bar):
????return?bar
with?concurrent.futures.ThreadPoolExecutor(max_workers=10)?as?executor:
????to_do?=?[]
????for?i?in?range(10):??#?模擬多個(gè)任務(wù)
????????future?=?executor.submit(foo,?f"hello?world!?{i}")
????????to_do.append(future)
????for?future?in?concurrent.futures.as_completed(to_do):??#?并發(fā)執(zhí)行
????????print(future.result())
某次運(yùn)行的結(jié)果如下:
hello world! 8
hello world! 3
hello world! 5
hello world! 2
hello world! 9
hello world! 7
hello world! 4
hello world! 0
hello world! 1
hello world! 6
最后的話
本文分享了獲取線程返回值的 3 種方法,推薦使用第三種
到此這篇關(guān)于詳解Python獲取線程返回值的三種方式的文章就介紹到這了,更多相關(guān)Python獲取線程返回值內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python 標(biāo)準(zhǔn)差計(jì)算的實(shí)現(xiàn)(std)
這篇文章主要介紹了python 標(biāo)準(zhǔn)差計(jì)算的實(shí)現(xiàn)(std),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07
Python利用redis-py實(shí)現(xiàn)集合與有序集合的常用指令操作
這篇文章我們將來(lái)學(xué)習(xí)?redis-py?這個(gè)模塊針對(duì)?“集合”?與?"有序集合"的一些常用指令操作,文中的示例代碼講解詳細(xì),需要的可以參考一下2022-09-09
python將excel轉(zhuǎn)換為csv的代碼方法總結(jié)
在本篇文章里小編給大家分享了關(guān)于python如何將excel轉(zhuǎn)換為csv的實(shí)例方法和代碼內(nèi)容,需要的朋友們學(xué)習(xí)下。2019-07-07
完美解決安裝完tensorflow后pip無(wú)法使用的問(wèn)題
今天小編就為大家分享一篇完美解決安裝完tensorflow后pip無(wú)法使用的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-06-06
在Python中處理字符串之isdecimal()方法的使用
這篇文章主要介紹了在Python中處理字符串之isdecimal()方法的使用,是Python入門學(xué)習(xí)的基礎(chǔ)知識(shí),需要的朋友可以參考下2015-05-05
Python實(shí)現(xiàn)按逗號(hào)分隔列表的方法
今天小編就為大家分享一篇Python實(shí)現(xiàn)按逗號(hào)分隔列表的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-10-10
python過(guò)濾字符串中不屬于指定集合中字符的類實(shí)例
這篇文章主要介紹了python過(guò)濾字符串中不屬于指定集合中字符的類,涉及Python針對(duì)字符串與集合的相關(guān)操作技巧,需要的朋友可以參考下2015-06-06
Python實(shí)現(xiàn)基于Fasttext的商品評(píng)論數(shù)據(jù)分類的操作流程
這篇文章主要介紹了Python實(shí)現(xiàn)基于Fasttext的商品評(píng)論數(shù)據(jù)分類,今天使用的fasttext更像是一個(gè)集成的庫(kù),把向量化和分類一起做掉了,這個(gè)對(duì)于使用層面來(lái)講就更方便了一些,需要的朋友可以參考下2022-06-06
python編寫(xiě)一個(gè)會(huì)算賬的腳本的示例代碼
這篇文章主要介紹了python編寫(xiě)一個(gè)會(huì)算賬的腳本,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-06-06

