淺談Python3多線程之間的執(zhí)行順序問(wèn)題
一個(gè)多線程的題:定義三個(gè)線程ID分別為ABC,每個(gè)線程打印10遍自己的線程ID,按ABCABC……的順序進(jìn)行打印輸出。
我的解法:
from threading import Thread, Lock # 由_acquire解鎖執(zhí)行后釋放_(tái)release鎖 def _print(_id: str, _acquire: Lock, _release: Lock) -> None: for i in range(10): _acquire.acquire() print(f"id:{_id}") _release.release() if __name__ == '__main__': # 創(chuàng)建三個(gè)鎖供3個(gè)線程使用 mutex1 = Lock() mutex2 = Lock() mutex3 = Lock() # 定義三個(gè)線程A、B、C # 線程A需要mutex1解鎖執(zhí)行后釋放mutex2 # 線程B需要mutex2解鎖執(zhí)行后釋放mutex3 # 線程C需要mutex3解鎖執(zhí)行后釋放mutex1 # 元組中第一位是自定義的線程ID,第二位是解鎖需要的鎖,第三位是釋放的鎖 threads = [Thread(target=_print, args=[i[0], i[1], i[2]]) for i in [('A', mutex1, mutex2), ('B', mutex2, mutex3), ('C', mutex3, mutex1)]] # 把mutex2和mutex3這兩把鎖先用了以便阻塞線程2和線程3的執(zhí)行 mutex2.acquire() mutex3.acquire() # 接下來(lái)只有線程A可以先執(zhí)行是因?yàn)閙utex1并沒(méi)有被占用 # 線程B和線程C需要分別等待著鎖2和鎖3的釋放才能繼續(xù)執(zhí)行 [thr.start() for thr in threads] [thr.join() for thr in threads]
補(bǔ)充知識(shí):python線程執(zhí)行代碼封裝和執(zhí)行順序
線程-注意點(diǎn)
1. 線程執(zhí)行代碼的封裝
通過(guò)上一小節(jié),能夠看出,通過(guò)使用threading模塊能完成多任務(wù)的程序開(kāi)發(fā),為了讓每個(gè)線程的封裝性更完美,所以使用threading模塊時(shí),往往會(huì)定義一個(gè)新的子類(lèi)class,只要繼承threading.Thread就可以了,然后重寫(xiě)run方法
示例如下:
#coding=utf-8 import threading import time class MyThread(threading.Thread): def run(self): for i in range(3): time.sleep(1) msg = "I'm "+self.name+' @ '+str(i) #name屬性中保存的是當(dāng)前線程的名字 print(msg) if __name__ == '__main__': t = MyThread() t.start()
說(shuō)明
python的threading.Thread類(lèi)有一個(gè)run方法,用于定義線程的功能函數(shù),可以在自己的線程類(lèi)中覆蓋該方法。而創(chuàng)建自己的線程實(shí)例后,通過(guò)Thread類(lèi)的start方法,可以啟動(dòng)該線程,交給python虛擬機(jī)進(jìn)行調(diào)度,當(dāng)該線程獲得執(zhí)行的機(jī)會(huì)時(shí),就會(huì)調(diào)用run方法執(zhí)行線程。
2. 線程的執(zhí)行順序
#coding=utf-8 import threading import time class MyThread(threading.Thread): def run(self): for i in range(3): time.sleep(1) msg = "I'm "+self.name+' @ '+str(i) print(msg) def test(): for i in range(5): t = MyThread() t.start() if __name__ == '__main__': test()
執(zhí)行結(jié)果:(運(yùn)行的結(jié)果可能不一樣,但是大體是一致的)
I'm Thread-1 @ 0 I'm Thread-2 @ 0 I'm Thread-5 @ 0 I'm Thread-3 @ 0 I'm Thread-4 @ 0 I'm Thread-3 @ 1 I'm Thread-4 @ 1 I'm Thread-5 @ 1 I'm Thread-1 @ 1 I'm Thread-2 @ 1 I'm Thread-4 @ 2 I'm Thread-5 @ 2 I'm Thread-2 @ 2 I'm Thread-1 @ 2 I'm Thread-3 @ 2
說(shuō)明
從代碼和執(zhí)行結(jié)果我們可以看出,多線程程序的執(zhí)行順序是不確定的。當(dāng)執(zhí)行到sleep語(yǔ)句時(shí),線程將被阻塞(Blocked),到sleep結(jié)束后,線程進(jìn)入就緒(Runnable)狀態(tài),等待調(diào)度。而線程調(diào)度將自行選擇一個(gè)線程執(zhí)行。上面的代碼中只能保證每個(gè)線程都運(yùn)行完整個(gè)run函數(shù),但是線程的啟動(dòng)順序、run函數(shù)中每次循環(huán)的執(zhí)行順序都不能確定。
3. 總結(jié)
每個(gè)線程默認(rèn)有一個(gè)名字,盡管上面的例子中沒(méi)有指定線程對(duì)象的name,但是python會(huì)自動(dòng)為線程指定一個(gè)名字。
當(dāng)線程的run()方法結(jié)束時(shí)該線程完成。
無(wú)法控制線程調(diào)度程序,但可以通過(guò)別的方式來(lái)影響線程調(diào)度的方式。
以上這篇淺談Python3多線程之間的執(zhí)行順序問(wèn)題就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- python多線程實(shí)現(xiàn)同時(shí)執(zhí)行兩個(gè)while循環(huán)的操作
- 解決python多線程報(bào)錯(cuò):AttributeError: Can''t pickle local object問(wèn)題
- 解決python ThreadPoolExecutor 線程池中的異常捕獲問(wèn)題
- Python mutiprocessing多線程池pool操作示例
- Python線程池模塊ThreadPoolExecutor用法分析
- python線程池threadpool實(shí)現(xiàn)篇
- 用Python實(shí)現(xiàn)一個(gè)簡(jiǎn)單的線程池
- Python 線程池模塊之多線程操作代碼
相關(guān)文章
在Ubuntu系統(tǒng)下安裝使用Python的GUI工具wxPython
這篇文章主要介紹了在Ubuntu系統(tǒng)下安裝使用Python的GUI工具wxPython的方法,wxPython可以為Python提供強(qiáng)大的圖形化界面開(kāi)發(fā)支持,需要的朋友可以參考下2016-02-02pycharm使用docker容器開(kāi)發(fā)的詳細(xì)教程
這篇文章主要介紹了pycharm使用docker容器開(kāi)發(fā)的詳細(xì)教程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-01-01Pytorch dataloader在加載最后一個(gè)batch時(shí)卡死的解決
這篇文章主要介紹了Pytorch dataloader在加載最后一個(gè)batch時(shí)卡死的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-05-05Python3操作SQL Server數(shù)據(jù)庫(kù)(實(shí)例講解)
下面小編就為大家?guī)?lái)一篇Python3操作SQL Server數(shù)據(jù)庫(kù)(實(shí)例講解)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-10-10詳解python3中tkinter知識(shí)點(diǎn)
本篇文章給大家分享了關(guān)于python3中tkinter的相關(guān)知識(shí)點(diǎn)以及實(shí)例代碼,有興趣的朋友參考下。2018-06-06vue.js實(shí)現(xiàn)輸入框輸入值內(nèi)容實(shí)時(shí)響應(yīng)變化示例
這篇文章主要介紹了vue.js實(shí)現(xiàn)輸入框輸入值內(nèi)容實(shí)時(shí)響應(yīng)變化,結(jié)合實(shí)例形式分析了vue.js使用v-model屬性進(jìn)行數(shù)據(jù)綁定的相關(guān)操作技巧,需要的朋友可以參考下2018-07-07python基礎(chǔ)編程小實(shí)例之計(jì)算圓的面積
Python是最常用的編程語(yǔ)言,這種語(yǔ)言就是一種可以快速開(kāi)發(fā)應(yīng)用的解釋型語(yǔ)言,有些用戶(hù)不知道該怎么在Python編程里計(jì)算圓的面積,現(xiàn)在就給大家具體解釋一下,下面這篇文章主要給大家介紹了關(guān)于python基礎(chǔ)編程小實(shí)例之計(jì)算圓的面積的相關(guān)資料,需要的朋友可以參考下2023-03-03