python 在threading中如何處理主進(jìn)程和子線程的關(guān)系
之前用python的多線程,總是處理不好進(jìn)程和線程之間的關(guān)系。后來發(fā)現(xiàn)了join和setDaemon函數(shù),才終于弄明白。下面總結(jié)一下。
1.使用join函數(shù)后,主進(jìn)程會(huì)在調(diào)用join的地方等待子線程結(jié)束,然后才接著往下執(zhí)行。
join使用實(shí)例如下:
import time
import random
import threading
class worker(threading.Thread):
def __init__(self):
threading.Thread.__init__(self)
def run(self):
t = random.randint(1,10)
time.sleep(t)
print "This is " + self.getName() + ";I sleep %d second."%(t)
tsk = []
for i in xrange(0,5):
time.sleep(0.1)
thread = worker()
thread.start()
tsk.append(thread)
for tt in tsk:
tt.join()
print "This is the end of main thread."
運(yùn)行結(jié)果如下:
# python testjoin.py This is Thread-3;I sleep 2 second. This is Thread-1;I sleep 4 second. This is Thread-2;I sleep 7 second. This is Thread-4;I sleep 7 second. This is Thread-5;I sleep 7 second. This is the end of main thread.
這里創(chuàng)建了5個(gè)子線程,每個(gè)線程隨機(jī)等待1-10秒后打印退出;主線程分別等待5個(gè)子線程結(jié)束。最后結(jié)果是先顯示各個(gè)子線程,再顯示主進(jìn)程的結(jié)果。
2. 如果使用的setDaemon函數(shù),則與join相反,主進(jìn)程結(jié)束的時(shí)候不會(huì)等待子線程。
setDaemon函數(shù)使用實(shí)例:
import time
import random
import threading
class worker(threading.Thread):
def __init__(self):
threading.Thread.__init__(self)
def run(self):
t = random.randint(1,10)
time.sleep(t)
print "This is " + self.getName() + ";I sleep %d second."%(t)
tsk = []
for i in xrange(0,5):
time.sleep(0.1)
thread = worker()
thread.setDaemon(True)
thread.start()
tsk.append(thread)
print "This is the end of main thread."
這里設(shè)置主進(jìn)程為守護(hù)進(jìn)程,當(dāng)主進(jìn)程結(jié)束的時(shí)候,子線程被中止
運(yùn)行結(jié)果如下:
#python testsetDaemon.py
This is the end of main thread.
3、如果沒有使用join和setDaemon函數(shù),則主進(jìn)程在創(chuàng)建子線程后,直接運(yùn)行后面的代碼,主程序一直掛起,直到子線程結(jié)束才能結(jié)束。
import time
import random
import threading
class worker(threading.Thread):
def __init__(self):
threading.Thread.__init__(self)
def run(self):
t = random.randint(1,10)
time.sleep(t)
print "This is " + self.getName() + ";I sleep %d second."%(t)
tsk = []
for i in xrange(0,5):
time.sleep(0.1)
thread = worker()
thread.start()
tsk.append(thread)
print "This is the end of main thread."
運(yùn)行結(jié)果如下:
# python testthread.py This is the end of main thread. This is Thread-4;I sleep 1 second. This is Thread-3;I sleep 7 second. This is Thread-5;I sleep 7 second. This is Thread-1;I sleep 10 second. This is Thread-2;I sleep 10 second.
補(bǔ)充知識(shí):Python Thread和Process對比
原因:進(jìn)程和線程的差距(方向不同,之針對這個(gè)實(shí)例)
# coding=utf-8
import logging
import multiprocessing
import os
import time
from threading import Thread
logging.basicConfig(
level=logging.INFO,
format="%(asctime)s 【 %(process)d 】 %(processName)s %(message)s"
)
def func (i):
# logging.info(f'子:{os.getpid()},\t{i}')
return f'子:{os.getpid()},\t{i}'
def main (ctx):
start01 = time.time()
ts = [Thread(target=func, args=(i,)) for i in range(100)]
[t.start() for t in ts]
[t.join() for t in ts]
end01 = time.time() - start01
logging.info(f"線程花費(fèi)的時(shí)間:{end01}秒")
start02 = time.time()
ps = [ctx.Process(target=func, args=(i,)) for i in range(100)]
[p.start() for p in ps]
[p.join() for p in ps]
end02 = time.time() - start02
logging.info(f"進(jìn)程花費(fèi)的時(shí)間:{end02}秒")
if __name__ == '__main__':
# windows 啟動(dòng)方式
multiprocessing.set_start_method('spawn')
# 獲取上下文
ctx = multiprocessing.get_context('spawn')
# 檢查這是否是凍結(jié)的可執(zhí)行文件中的偽分支進(jìn)程。
ctx.freeze_support()
main(ctx)
輸出:
2019-10-06 14:17:22,729 【 7412 】 MainProcess 線程花費(fèi)的時(shí)間:0.012967586517333984秒
2019-10-06 14:17:25,671 【 7412 】 MainProcess 進(jìn)程花費(fèi)的時(shí)間:2.9418249130249023秒
以上這篇python 在threading中如何處理主進(jìn)程和子線程的關(guān)系就是小編分享給大家的全部內(nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Django admin model 漢化顯示文字的實(shí)現(xiàn)方法
今天小編就為大家分享一篇Django admin model 漢化顯示文字的實(shí)現(xiàn)方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-08-08
Python?運(yùn)算符Inplace?與Standard?
這篇文章主要介紹了Python?運(yùn)算符Inplace?與Standard,nplace運(yùn)算符的行為類似于普通運(yùn)算符,只是它們在可變和不可變目標(biāo)的情況下以不同的方式運(yùn)行2022-09-09
Python利用PySimpleGUI實(shí)現(xiàn)自制桌面翻譯神器
工作上經(jīng)常需要與外國友人郵件溝通,奈何工作電腦沒有安裝有道詞典一類的翻譯軟件,結(jié)合自己的需要,自己用PySimpleGUI擼一個(gè)桌面翻譯神器,感興趣的可以了解一下2022-09-09
Python使用matplotlib實(shí)現(xiàn)交換式圖形顯示功能示例
這篇文章主要介紹了Python使用matplotlib實(shí)現(xiàn)交換式圖形顯示功能,結(jié)合實(shí)例形式詳細(xì)分析了Python基于matplotlib模塊的數(shù)值運(yùn)算與圖形繪制相關(guān)操作技巧,需要的朋友可以參考下2019-09-09
Python中用append()連接后多出一列Unnamed的解決
Python中用append()連接后多出一列Unnamed的解決方案,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-01-01
Numpy中創(chuàng)建數(shù)組的9種方式小結(jié)
本文主要介紹了Numpy中創(chuàng)建數(shù)組的9種方式小結(jié),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03
Python八大常見排序算法定義、實(shí)現(xiàn)及時(shí)間消耗效率分析
這篇文章主要介紹了Python八大常見排序算法定義、實(shí)現(xiàn)及時(shí)間消耗效率分析,結(jié)合具體實(shí)例形式對比分析了冒泡排序、直接插入排序、選擇排序、歸并排序、希爾排序、桶排序、堆排序等排序算法的使用與執(zhí)行效率,需要的朋友可以參考下2018-04-04

