电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

分析Python感知線程狀態的解決方案之Event與信號量

瀏覽:5日期:2022-06-16 16:01:12
目錄一、停止線程二、線程信號的傳遞三、信號量四、總結一、停止線程

利用Threading庫我們可以很方便地創建線程,讓它按照我們的想法執行我們想讓它執行的事情,從而加快程序運行的效率。然而有一點坑爹的是,線程創建之后,就交給了操作系統執行,我們無法直接結束一個線程,也無法給它發送信號,無法調整它的調度,也沒有其他高級操作。如果想要相關的功能,只能自己開發。

怎么開發呢?

我們創建線程的時候指定了target等于一個我們想讓它執行的函數,這個函數并不一定是全局函數,實際上也可以是一個對象中的函數。如果是對象中的函數,那么我們就可以在這個函數當中獲取到對象中的其他信息,我們可以利用這一點來實現手動控制線程的停止。

說起來好像不太好理解,但是看下代碼真的非常簡單:

import timefrom threading import Threadclass TaskWithSwitch: def __init__(self):self._running = True def terminate(self):self._running = False def run(self, n):while self._running and n > 0: print(’Running {}’.format(n)) n -= 1 time.sleep(1)c = TaskWithSwitch()t = Thread(target=c.run, args=(10, ))t.start()c.terminate()t.join()

如果你運行這段代碼,會發現屏幕上只輸出了10,因為我們將_running這個字段置為False之后,下次循環的時候不再滿足循環條件,它就會自己退出了。

分析Python感知線程狀態的解決方案之Event與信號量

如果我們想要用多線程來讀取IO,由于IO可能存在堵塞,所以可能會出現線程一直無法返回的情況。也就是說我們在循環內部卡死了,這個時候單純用_running來判斷還是不夠的,我們需要在線程內部設置計時器,防止循環內部的卡死。

class IOTask: def __init__(self):self._running = True def terminate(self):self._running = False def run(self, sock):# 在socket中設置計時器sock.settimeout(10)while self._running: try:# 由于設置了計時器,所以這里不會永久等待data = sock.recv(1024)break except socket.timeout:continuereturn二、線程信號的傳遞

我們之所以如此費勁才能控制線程的運行,主要原因是線程的狀態是不可知的,并且我們無法直接操作它,因為它是被操作系統管理的。我們運行的主線程和創建出來的線程是獨立的,兩者之間并沒有從屬關系,所以想要實現對線程的狀態進行控制,往往需要我們通過其他手段來實現。

我們來思考一個場景,假設我們有一個任務,需要在另外一個線程運行結束之后才能開始執行。要想要實現這一點,就必須對線程的狀態有所感知,需要其他線程傳遞出信號來才行。我們可以使用threading中的Event工具來實現這一點。Event工具就是可以用來傳遞信號的,就好像是一個開關,當一個線程執行完成之后,會去啟動這個開關。而這個開關控制著另外一段邏輯的運行。

我們來看下樣例代碼:

import timefrom threading import Thread, Eventdef run_in_thread(): time.sleep(1) print(’Thread is running’)t = Thread(target=run_in_thread)t.start()print(’Main thread print’)

我們在線程里面就只做了輸出一行提示符,沒有其他任何邏輯。由于我們在run_in_thread函數當中沉睡了1s,所以一定是先輸出Main thread print再輸出的Thread is running。假設這個線程是一個很重要的任務,我們希望主線程能夠等待它運行到一個階段再往下執行,我們應該怎么辦呢?

注意,這里說的是運行到一個階段,并不是運行結束。運行結束我們很好處理,可以通過join來完成。但如果不是運行結束,而是運行完成了某一個階段,當然通過join也可以,但是會損害整體的效率。這個時候我們就必須要用上Event了。加上Event之后,我們再來看下代碼:

import timefrom threading import Thread, Eventdef run_in_thread(event): time.sleep(1) print(’Thread is running’) # set一下event,這樣外面wait的部分就會被啟動 event.set()# 初始化Eventevent = Event()t = Thread(target=run_in_thread, args=(event, ))t.start()# event等待setevent.wait()print(’Main thread print’)

整體的邏輯沒有太多的修改,主要的是增加了幾行關于Event的使用代碼。

我們如果要用到Event,最好在代碼當中只使用一次。當然通過Event中的clear方法我們可以重置Event的值,但問題是我們沒辦法保證重置的這個邏輯會在wait之前執行。如果是在之后執行的,那么就會問題,并且在debug的時候會異常痛苦,因為bug不是必現的,而是有時候會出現有時候不會出現。這種情況往往都是因為多線程的使用問題。

所以如果要多次使用開關和信號的話,不要使用Event,可以使用信號量。

三、信號量

Event的問題在于如果多個線程在等待Event的發生,當它一旦被set的時候,那么這些線程都會同時執行。但有時候我們并不希望這樣,我們希望可以控制這些線程一個一個地運行。如果想要做到這一點,Event就無法滿足了,而需要使用信號量。

信號量和Event的使用方法類似,不同的是,信號量可以保證每次只會啟動一個線程。因為這兩者的底層邏輯不太一致,對于Event來說,它更像是一個開關。一旦開關啟動,所有和這個開關關聯的邏輯都會同時執行。而信號量則像是許可證,只有拿到許可證的線程才能執行工作,并且許可證一次只發一張。

想要使用信號量并不需要自己開發,thread庫當中為我們提供了現成的工具——Semaphore,我們來看它的使用代碼:

# 工作線程def worker(n, sema): # 等待信號量 sema.acquire() print(’Working’, n)# 初始化sema = threading.Semaphore(0)nworkers = 10for n in range(nworkers): t = threading.Thread(target=worker, args=(n, sema,)) t.start()

在上面的代碼當中我們創建了10個線程,雖然這些線程都被啟動了,但是都不會執行邏輯,因為sema.acquire是一個阻塞方法,沒有監聽到信號量是會一直掛起等待。

分析Python感知線程狀態的解決方案之Event與信號量

當我們釋放信號量之后,線程被啟動,才開始了執行。我們每釋放一個信號,則會多啟動一個線程。這里面的邏輯應該不難理解。

四、總結

在并發場景當中,多線程的使用絕不是多啟動幾個線程做不同的任務而已,我們需要線程間協作,需要同步、獲取它們的狀態,這是非常不容易的。一不小心就會出現幽靈bug,時顯時隱,這也是并發問題讓人頭疼的主要原因。

以上就是分析Python感知線程狀態的解決方案之Event與信號量的詳細內容,更多關于Python 感知線程狀態 Event與信號量的資料請關注好吧啦網其它相關文章!

標簽: Python 編程
相關文章:
主站蜘蛛池模板: 食品机械专用传感器-落料放大器-低价接近开关-菲德自控技术(天津)有限公司 | 品牌广告服务平台,好排名,好流量,好生意。 | 贵州科比特-防雷公司厂家提供贵州防雷工程,防雷检测,防雷接地,防雷设备价格,防雷产品报价服务-贵州防雷检测公司 | 光伏家 - 太阳能光伏发电_分布式光伏发电_太阳能光伏网 | 佛山市钱丰金属不锈钢蜂窝板定制厂家|不锈钢装饰线条|不锈钢屏风| 电梯装饰板|不锈钢蜂窝板不锈钢工艺板材厂家佛山市钱丰金属制品有限公司 | 火锅加盟_四川成都火锅店加盟_中国火锅连锁品牌十强_朝天门火锅【官网】 | 数码管_LED贴片灯_LED数码管厂家-无锡市冠卓电子科技有限公司 | 变频器维修公司_plc维修_伺服驱动器维修_工控机维修 - 夫唯科技 变位机,焊接变位机,焊接变位器,小型变位机,小型焊接变位机-济南上弘机电设备有限公司 | 六自由度平台_六自由度运动平台_三自由度摇摆台—南京全控科技 | 网站建设-高端品牌网站设计制作一站式定制_杭州APP/微信小程序开发运营-鼎易科技 | 不锈钢水管-不锈钢燃气管-卫生级不锈钢管件-不锈钢食品级水管-广东双兴新材料集团有限公司 | 裹包机|裹膜机|缠膜机|绕膜机-上海晏陵智能设备有限公司 | 小威小说网 - 新小威小说网 - 小威小说网小说搜索引擎 | 杭州顺源过滤机械有限公司官网-压滤机_板框压滤机_厢式隔膜压滤机厂家 | 武汉天安盾电子设备有限公司 - 安盾安检,武汉安检门,武汉安检机,武汉金属探测器,武汉测温安检门,武汉X光行李安检机,武汉防爆罐,武汉车底安全检查,武汉液体探测仪,武汉安检防爆设备 | 西点培训学校_法式西点培训班_西点师培训_西点蛋糕培训-广州烘趣西点烘焙培训学院 | 扬子叉车厂家_升降平台_电动搬运车|堆高车-扬子仓储叉车官网 | 十字轴_十字轴万向节_十字轴总成-南京万传机械有限公司 | 悬浮拼装地板_幼儿园_篮球场_悬浮拼接地板-山东悬浮拼装地板厂家 | 灌装封尾机_胶水灌装机_软管灌装封尾机_无锡和博自动化机械制造有限公司 | Brotu | 关注AI,Web3.0,VR/AR,GPT,元宇宙区块链数字产业 | 连栋温室大棚建造厂家-智能玻璃温室-薄膜温室_青州市亿诚农业科技 | 领袖户外_深度旅游、摄影旅游、小团慢旅行、驴友网 | 欧美日韩国产一区二区三区不_久久久久国产精品无码不卡_亚洲欧洲美洲无码精品AV_精品一区美女视频_日韩黄色性爱一级视频_日本五十路人妻斩_国产99视频免费精品是看4_亚洲中文字幕无码一二三四区_国产小萍萍挤奶喷奶水_亚洲另类精品无码在线一区 | 踏板力计,制动仪,非接触多功能速度仪,逆反射系数测试仪-创宇 | 外贮压-柜式-悬挂式-七氟丙烷-灭火器-灭火系统-药剂-价格-厂家-IG541-混合气体-贮压-非贮压-超细干粉-自动-灭火装置-气体灭火设备-探火管灭火厂家-东莞汇建消防科技有限公司 | 办公室家具_板式办公家具定制厂家-FMARTS福玛仕办公家具 | 净化车间装修_合肥厂房无尘室设计_合肥工厂洁净工程装修公司-安徽盛世和居装饰 | 上海洗地机-洗地机厂家-全自动洗地机-手推式洗地机-上海滢皓洗地机 | ph计,实验室ph计,台式ph计,实验室酸度计,台式酸度计 | 上海刑事律师|刑事辩护律师|专业刑事犯罪辩护律师免费咨询-[尤辰荣]金牌上海刑事律师团队 | 光环国际-新三板公司_股票代码:838504 | 土壤检测仪器_行星式球磨仪_土壤团粒分析仪厂家_山东莱恩德智能科技有限公司 | 西装定制/做厂家/公司_西装订做/制价格/费用-北京圣达信西装 | 储气罐,真空罐,缓冲罐,隔膜气压罐厂家批发价格,空压机储气罐规格型号-上海申容压力容器集团有限公司 | 蒸压釜-陶粒板隔墙板蒸压釜-山东鑫泰鑫智能装备有限公司 | 胀套-锁紧盘-风电锁紧盘-蛇形联轴器「厂家」-瑞安市宝德隆机械配件有限公司 | 沈阳建筑设计公司_加固改造设计_厂房设计_设计资质加盟【金辉设计】 | 英国雷迪地下管线探测仪-雷迪RD8100管线仪-多功能数字听漏仪-北京迪瑞进创科技有限公司 | 钢衬玻璃厂家,钢衬玻璃管道 -山东东兴扬防腐设备有限公司 | 郑州宣传片拍摄-TVC广告片拍摄-微电影短视频制作-河南优柿文化传媒有限公司 |