淺談Python3多線程之間的執(zhí)行順序問題
一個(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()
# 接下來只有線程A可以先執(zhí)行是因?yàn)閙utex1并沒有被占用
# 線程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í)行代碼的封裝
通過上一小節(jié),能夠看出,通過使用threading模塊能完成多任務(wù)的程序開發(fā),為了讓每個(gè)線程的封裝性更完美,所以使用threading模塊時(shí),往往會(huì)定義一個(gè)新的子類class,只要繼承threading.Thread就可以了,然后重寫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()
說明
python的threading.Thread類有一個(gè)run方法,用于定義線程的功能函數(shù),可以在自己的線程類中覆蓋該方法。而創(chuàng)建自己的線程實(shí)例后,通過Thread類的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
說明
從代碼和執(zhí)行結(jié)果我們可以看出,多線程程序的執(zhí)行順序是不確定的。當(dāng)執(zhí)行到sleep語句時(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è)名字,盡管上面的例子中沒有指定線程對(duì)象的name,但是python會(huì)自動(dòng)為線程指定一個(gè)名字。
當(dāng)線程的run()方法結(jié)束時(shí)該線程完成。
無法控制線程調(diào)度程序,但可以通過別的方式來影響線程調(diào)度的方式。
以上這篇淺談Python3多線程之間的執(zhí)行順序問題就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Python標(biāo)準(zhǔn)庫中隱藏的利器(示例詳解)
在命令行中直接使用Python標(biāo)準(zhǔn)庫的模塊,最大的好處就是就是不用寫代碼,就能使用其中的功能,當(dāng)臨時(shí)需要一些某些功能的時(shí)候,用這種方式會(huì)快捷,方便很多,這篇文章主要介紹了Python標(biāo)準(zhǔn)庫中隱藏的利器,需要的朋友可以參考下2023-11-11
python做http代理請(qǐng)求的項(xiàng)目實(shí)踐
本文主要介紹了使用Python Flask實(shí)現(xiàn)HTTP代理服務(wù)器的解決方案,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2025-03-03
基于PyQt5實(shí)現(xiàn)圖轉(zhuǎn)文功能(示例代碼)
PyQt提供了一個(gè)設(shè)計(jì)良好的窗口控件集合,具有更方便的操作性。學(xué)過VB的同學(xué)會(huì)知道,相比與VB的使用,在界面設(shè)計(jì)上元素更豐富,這篇文章主要介紹了基于PyQt5完成的圖轉(zhuǎn)文功能,需要的朋友可以參考下2022-06-06
使用django-crontab實(shí)現(xiàn)定時(shí)任務(wù)的示例
這篇文章主要介紹了使用django-crontab實(shí)現(xiàn)定時(shí)任務(wù),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-02-02
python使用opencv對(duì)圖像添加噪聲(高斯/椒鹽/泊松/斑點(diǎn))
這篇文章主要介紹了python使用opencv對(duì)圖像添加噪聲(高斯/椒鹽/泊松/斑點(diǎn)),具有一定的學(xué)習(xí)價(jià)值,需要的小伙伴可以參考一下,希望對(duì)你有所幫助2022-04-04
python 如何快速找出兩個(gè)電子表中數(shù)據(jù)的差異
下面小編就為大家?guī)硪黄猵ython 如何快速找出兩個(gè)電子表中數(shù)據(jù)的差異。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-05-05
Django同時(shí)連接多種數(shù)據(jù)庫的實(shí)現(xiàn)
在開發(fā)Django項(xiàng)目的時(shí)候,很多時(shí)候都是使用一個(gè)數(shù)據(jù)庫,即settings 中只有default數(shù)據(jù)庫,但是有一些項(xiàng)目確實(shí)也需要使用多個(gè)數(shù)據(jù)庫,本文主要介紹了Django同時(shí)連接多種數(shù)據(jù)庫的實(shí)現(xiàn),感興趣的可以了解一下2023-11-11

