日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

分析Python感知線程狀態(tài)的解決方案之Event與信號(hào)量

瀏覽:19日期:2022-06-16 16:01:12
目錄一、停止線程二、線程信號(hào)的傳遞三、信號(hào)量四、總結(jié)一、停止線程

利用Threading庫(kù)我們可以很方便地創(chuàng)建線程,讓它按照我們的想法執(zhí)行我們想讓它執(zhí)行的事情,從而加快程序運(yùn)行的效率。然而有一點(diǎn)坑爹的是,線程創(chuàng)建之后,就交給了操作系統(tǒng)執(zhí)行,我們無法直接結(jié)束一個(gè)線程,也無法給它發(fā)送信號(hào),無法調(diào)整它的調(diào)度,也沒有其他高級(jí)操作。如果想要相關(guān)的功能,只能自己開發(fā)。

怎么開發(fā)呢?

我們創(chuàng)建線程的時(shí)候指定了target等于一個(gè)我們想讓它執(zhí)行的函數(shù),這個(gè)函數(shù)并不一定是全局函數(shù),實(shí)際上也可以是一個(gè)對(duì)象中的函數(shù)。如果是對(duì)象中的函數(shù),那么我們就可以在這個(gè)函數(shù)當(dāng)中獲取到對(duì)象中的其他信息,我們可以利用這一點(diǎn)來實(shí)現(xiàn)手動(dòng)控制線程的停止。

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

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()

如果你運(yùn)行這段代碼,會(huì)發(fā)現(xiàn)屏幕上只輸出了10,因?yàn)槲覀儗running這個(gè)字段置為False之后,下次循環(huán)的時(shí)候不再滿足循環(huán)條件,它就會(huì)自己退出了。

分析Python感知線程狀態(tài)的解決方案之Event與信號(hào)量

如果我們想要用多線程來讀取IO,由于IO可能存在堵塞,所以可能會(huì)出現(xiàn)線程一直無法返回的情況。也就是說我們?cè)谘h(huán)內(nèi)部卡死了,這個(gè)時(shí)候單純用_running來判斷還是不夠的,我們需要在線程內(nèi)部設(shè)置計(jì)時(shí)器,防止循環(huán)內(nèi)部的卡死。

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

我們之所以如此費(fèi)勁才能控制線程的運(yùn)行,主要原因是線程的狀態(tài)是不可知的,并且我們無法直接操作它,因?yàn)樗潜徊僮飨到y(tǒng)管理的。我們運(yùn)行的主線程和創(chuàng)建出來的線程是獨(dú)立的,兩者之間并沒有從屬關(guān)系,所以想要實(shí)現(xiàn)對(duì)線程的狀態(tài)進(jìn)行控制,往往需要我們通過其他手段來實(shí)現(xiàn)。

我們來思考一個(gè)場(chǎng)景,假設(shè)我們有一個(gè)任務(wù),需要在另外一個(gè)線程運(yùn)行結(jié)束之后才能開始執(zhí)行。要想要實(shí)現(xiàn)這一點(diǎn),就必須對(duì)線程的狀態(tài)有所感知,需要其他線程傳遞出信號(hào)來才行。我們可以使用threading中的Event工具來實(shí)現(xiàn)這一點(diǎn)。Event工具就是可以用來傳遞信號(hào)的,就好像是一個(gè)開關(guān),當(dāng)一個(gè)線程執(zhí)行完成之后,會(huì)去啟動(dòng)這個(gè)開關(guān)。而這個(gè)開關(guān)控制著另外一段邏輯的運(yùn)行。

我們來看下樣例代碼:

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’)

我們?cè)诰€程里面就只做了輸出一行提示符,沒有其他任何邏輯。由于我們?cè)趓un_in_thread函數(shù)當(dāng)中沉睡了1s,所以一定是先輸出Main thread print再輸出的Thread is running。假設(shè)這個(gè)線程是一個(gè)很重要的任務(wù),我們希望主線程能夠等待它運(yùn)行到一個(gè)階段再往下執(zhí)行,我們應(yīng)該怎么辦呢?

注意,這里說的是運(yùn)行到一個(gè)階段,并不是運(yùn)行結(jié)束。運(yùn)行結(jié)束我們很好處理,可以通過join來完成。但如果不是運(yùn)行結(jié)束,而是運(yùn)行完成了某一個(gè)階段,當(dāng)然通過join也可以,但是會(huì)損害整體的效率。這個(gè)時(shí)候我們就必須要用上Event了。加上Event之后,我們?cè)賮砜聪麓a:

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

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

我們?nèi)绻玫紼vent,最好在代碼當(dāng)中只使用一次。當(dāng)然通過Event中的clear方法我們可以重置Event的值,但問題是我們沒辦法保證重置的這個(gè)邏輯會(huì)在wait之前執(zhí)行。如果是在之后執(zhí)行的,那么就會(huì)問題,并且在debug的時(shí)候會(huì)異常痛苦,因?yàn)閎ug不是必現(xiàn)的,而是有時(shí)候會(huì)出現(xiàn)有時(shí)候不會(huì)出現(xiàn)。這種情況往往都是因?yàn)槎嗑€程的使用問題。

所以如果要多次使用開關(guān)和信號(hào)的話,不要使用Event,可以使用信號(hào)量。

三、信號(hào)量

Event的問題在于如果多個(gè)線程在等待Event的發(fā)生,當(dāng)它一旦被set的時(shí)候,那么這些線程都會(huì)同時(shí)執(zhí)行。但有時(shí)候我們并不希望這樣,我們希望可以控制這些線程一個(gè)一個(gè)地運(yùn)行。如果想要做到這一點(diǎn),Event就無法滿足了,而需要使用信號(hào)量。

信號(hào)量和Event的使用方法類似,不同的是,信號(hào)量可以保證每次只會(huì)啟動(dòng)一個(gè)線程。因?yàn)檫@兩者的底層邏輯不太一致,對(duì)于Event來說,它更像是一個(gè)開關(guān)。一旦開關(guān)啟動(dòng),所有和這個(gè)開關(guān)關(guān)聯(lián)的邏輯都會(huì)同時(shí)執(zhí)行。而信號(hào)量則像是許可證,只有拿到許可證的線程才能執(zhí)行工作,并且許可證一次只發(fā)一張。

想要使用信號(hào)量并不需要自己開發(fā),thread庫(kù)當(dāng)中為我們提供了現(xiàn)成的工具——Semaphore,我們來看它的使用代碼:

# 工作線程def worker(n, sema): # 等待信號(hào)量 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()

在上面的代碼當(dāng)中我們創(chuàng)建了10個(gè)線程,雖然這些線程都被啟動(dòng)了,但是都不會(huì)執(zhí)行邏輯,因?yàn)閟ema.acquire是一個(gè)阻塞方法,沒有監(jiān)聽到信號(hào)量是會(huì)一直掛起等待。

分析Python感知線程狀態(tài)的解決方案之Event與信號(hào)量

當(dāng)我們釋放信號(hào)量之后,線程被啟動(dòng),才開始了執(zhí)行。我們每釋放一個(gè)信號(hào),則會(huì)多啟動(dòng)一個(gè)線程。這里面的邏輯應(yīng)該不難理解。

四、總結(jié)

在并發(fā)場(chǎng)景當(dāng)中,多線程的使用絕不是多啟動(dòng)幾個(gè)線程做不同的任務(wù)而已,我們需要線程間協(xié)作,需要同步、獲取它們的狀態(tài),這是非常不容易的。一不小心就會(huì)出現(xiàn)幽靈bug,時(shí)顯時(shí)隱,這也是并發(fā)問題讓人頭疼的主要原因。

以上就是分析Python感知線程狀態(tài)的解決方案之Event與信號(hào)量的詳細(xì)內(nèi)容,更多關(guān)于Python 感知線程狀態(tài) Event與信號(hào)量的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Python 編程
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产精品一级| 鲁大师精品99久久久| 国产劲爆久久| 欧美色综合网| 欧美日韩中文| 久久只有精品| 日韩在线电影| 日本不卡不码高清免费观看| 三级欧美韩日大片在线看| 国产精品人人爽人人做我的可爱| 日韩视频不卡| 蜜臀a∨国产成人精品| 一区二区三区网站| 91精品一区| 国产精品中文字幕制服诱惑| 国产伦精品一区二区三区千人斩| 久久超碰99| 波多野结衣久久精品| 久久激情一区| 性色av一区二区怡红| 日韩精品五月天| 日本精品国产| 久久精品理论片| 欧美aa在线观看| 午夜电影亚洲| 日韩国产一二三区| 黄色精品视频| 午夜久久99| 88久久精品| 色婷婷色综合| 好吊一区二区三区| 91欧美精品| 日韩免费久久| 免费美女久久99| 国产精品jk白丝蜜臀av小说| 日韩一区三区| 免费视频一区三区| 亚洲精品一区二区在线播放∴| 日本aⅴ免费视频一区二区三区| 久久99视频| 香蕉国产精品| 国产精选久久| 婷婷综合亚洲| 国产精品久久| 欧美精品一区二区久久| 日韩国产欧美三级| 青青久久av| 日韩在线麻豆| 岛国av在线播放| 老鸭窝亚洲一区二区三区| 国产精选久久| 日韩网站中文字幕| 日韩一二三区在线观看| bbw在线视频| 日韩欧美在线精品| 中文字幕在线免费观看视频| 中文字幕一区二区三区日韩精品| 精品久久精品| 亚洲免费观看高清完整版在线观| 成人在线黄色| 日韩av中文字幕一区二区| 99久久精品费精品国产| 青青草91视频| 婷婷综合社区| 日本久久综合| 欧美亚洲一级| 欧美69视频| 欧美片第1页综合| 日韩亚洲国产欧美| www.九色在线| 国产精品magnet| 日韩中文字幕一区二区高清99| 91精品精品| 久久影院一区二区三区| 亚洲乱码一区| 日韩视频不卡| 在线日韩电影| 日韩欧美午夜| 国产精品2023| 亚洲精品日韩久久| 黄色av一区| 日韩欧美网址| 精品理论电影在线| 91大神在线观看线路一区| 一本一道久久a久久| 精品91久久久久| 久久免费高清| 日韩视频网站在线观看| 精品伊人久久| 国产精品毛片视频| 日韩高清成人在线| 亚洲精品免费观看| 国产精品婷婷| 亚洲国产一区二区三区在线播放| 秋霞影视一区二区三区| 中文字幕在线视频久| 国产精品香蕉| 国产精品一区二区美女视频免费看| 日本不卡视频在线观看| 亚洲精品极品| 亚洲日韩中文字幕一区| 视频一区在线播放| 国产精品婷婷| 蜜臀久久久久久久| 亚洲无线观看| 婷婷视频一区二区三区| 亚洲我射av| 日本a级不卡| 国产精品亚洲四区在线观看 | sm捆绑调教国产免费网站在线观看| 麻豆国产91在线播放| 国产极品嫩模在线观看91精品| 国产精品一区二区免费福利视频| 国产精品一区亚洲| 国产精品22p| 丁香婷婷久久| 三级在线看中文字幕完整版| 亚洲黄色免费av| 999久久久免费精品国产| 在线日韩欧美| 玖玖精品视频| 日本中文字幕一区二区| 日本伊人午夜精品| 91精品丝袜国产高跟在线| 91精品丝袜国产高跟在线| 欧美成人精品午夜一区二区| 国内精品亚洲| 亚洲午夜电影| 中文字幕日本一区二区| 日韩1区2区日韩1区2区| 国产精品成人a在线观看| 精品欧美一区二区三区在线观看| 亚洲精品91| 欧美日韩va| 黄色aa久久| 亚洲在线网站| 91综合久久爱com| 日韩大片在线| 日韩视频一区| 国产精品亚洲人成在99www| 婷婷激情一区| 日韩制服丝袜先锋影音| 国产伦一区二区三区| 美女网站视频一区| 免费欧美在线视频| 久久精品国产999大香线蕉| 日韩成人亚洲| 亚洲精品一级二级三级| 国产一区精品福利| 欧美日韩四区| 国产乱人伦精品一区| 久久国产免费| 久久国产视频网| 久久精品理论片| 亚洲一卡久久| 美腿丝袜在线亚洲一区| 欧美亚洲激情| 国产精品大片免费观看| 欧美+亚洲+精品+三区| 欧美另类中文字幕| 婷婷亚洲五月色综合| 久久国际精品| 亚洲高清成人| 国产精品片aa在线观看| 欧美成人综合| 你懂的网址国产 欧美| 91精品1区| 日韩1区2区| 亚洲69av| 一区在线观看| 久久久精品久久久久久96| 国产精品任我爽爆在线播放| 性色av一区二区怡红| 日韩成人亚洲| 国产在线一区不卡| 久久国内精品自在自线400部| 视频一区视频二区中文| 色偷偷色偷偷色偷偷在线视频| 国产精品一卡| 7m精品国产导航在线| 亚洲影院天堂中文av色| 精品一区免费| 欧美黄色网页| 精品久久久亚洲| 国产精品3区| 久久国产日韩欧美精品| 亚洲人成网站在线在线观看| аⅴ资源天堂资源库在线| 日韩激情综合| 久久国产精品毛片| 91精品国产自产观看在线 | 亚洲高清毛片| 美女性感视频久久| 视频一区在线播放| 日韩视频在线一区二区三区 | 久久久久久夜| 久久精品资源| 少妇精品久久久一区二区| 成人看片网站| 里番精品3d一二三区|