網(wǎng)站首頁 編程語言 正文
提到線程,你的大腦應(yīng)該有這樣的印象:我們可以控制它何時開始,卻無法控制它何時結(jié)束,那么如何獲取線程的返回值呢?今天就分享一下自己的一些做法。
方法一
使用全局變量的列表,來保存返回值
ret_values?=?[]
def?thread_func(*args):
????...
????value?=?...
????ret_values.append(value)
選擇列表的一個原因是:列表的 append() 方法是線程安全的,CPython 中,GIL 防止對它們的并發(fā)訪問。如果你使用自定義的數(shù)據(jù)結(jié)構(gòu),在并發(fā)修改數(shù)據(jù)的地方需要加線程鎖。
如果事先知道有多少個線程,可以定義一個固定長度的列表,然后根據(jù)索引來存放返回值,比如:
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))
方法二
重寫 Thread 的 join 方法,返回線程函數(shù)的返回值
默認的 thread.join() 方法只是等待線程函數(shù)結(jié)束,沒有返回值,我們可以在此處返回函數(shù)的運行結(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())?#?此處會打印 hello world。
這樣當我們調(diào)用 thread.join() 等待線程結(jié)束的時候,也就得到了線程的返回值。
方法三
使用標準庫 concurrent.futures
我覺得前兩種方式實在太低級了,Python 的標準庫 concurrent.futures 提供更高級的線程操作,可以直接獲取線程的返回值,相當優(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):??#?模擬多個任務(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())
某次運行的結(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 種方法,推薦使用第三種
原文鏈接:https://mp.weixin.qq.com/s/SB7D_FkCOunUYgXaWM-ozw
相關(guān)推薦
- 2022-06-30 基于Python使用永中文檔轉(zhuǎn)換服務(wù)的方式_python
- 2022-04-29 C++?雙向循環(huán)鏈表類模版實例詳解_C 語言
- 2023-04-13 微信分享,wx.config參數(shù)設(shè)置
- 2022-10-03 Golang?Http請求返回結(jié)果處理_Golang
- 2022-02-01 Axure谷歌瀏覽器Chrome擴展程序下載及安裝方法
- 2022-12-22 C語言中pow函數(shù)使用方法、注意事項以及常見報錯原因_C 語言
- 2022-05-08 繼docker之后podman容器技術(shù)崛起_docker
- 2024-02-01 springboot @spring.active@啟動報錯
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支