Python多進(jìn)程并發(fā)與同步機(jī)制超詳細(xì)講解
在《多線程與同步》中介紹了多線程及存在的問題,而通過使用多進(jìn)程而非線程可有效地繞過全局解釋器鎖。 因此,通過multiprocessing模塊可充分地利用多核CPU的資源。
多進(jìn)程
多進(jìn)程是通過multiprocessing包來實(shí)現(xiàn)的,multiprocessing.Process對象(和多線程的threading.Thread類似)用來創(chuàng)建一個進(jìn)程對象:
- 在類UNIX平臺上,需要對每個Process對象調(diào)用join()方法 (實(shí)際上等同于wait)避免其成為僵尸進(jìn)程。
- multiprocessing提供了threading包中沒有的IPC(比如Pipe和Queue),效率上更高。應(yīng)優(yōu)先考慮Pipe和Queue,避免使用Lock/Event/Semaphore/Condition等同步方式。
- 多進(jìn)程應(yīng)盡量避免共享資源。必要時可以通過共享內(nèi)存和Manager的方法來共享資源。
僵尸進(jìn)程
在unix或unix-like系統(tǒng)中,當(dāng)一個子進(jìn)程退出后,它就會變成一個僵尸進(jìn)程,如果父進(jìn)程沒有通過wait系統(tǒng)調(diào)用來讀取這個子進(jìn)程的退出狀態(tài)的話,這個子進(jìn)程就會一直維持僵尸進(jìn)程狀態(tài)(占據(jù)部分系統(tǒng)資源,無法釋放)。
要清除僵尸進(jìn)程,有:
結(jié)束父進(jìn)程(一般是主進(jìn)程):當(dāng)父進(jìn)程退出的時候僵尸進(jìn)程也會被隨之清除。
讀取子進(jìn)程退出狀態(tài):如通過multiprocessing.Process產(chǎn)出的進(jìn)程可以:
- 調(diào)用join()來等待子進(jìn)程的方法來(內(nèi)部會wait子進(jìn)程);
- 在父進(jìn)程中處理SIGCHLD信號:在處理程序中調(diào)用wait系統(tǒng)調(diào)用或者直接設(shè)置為SIG_IGN來清除僵尸進(jìn)程;
把進(jìn)程變成孤兒進(jìn)程,這樣進(jìn)程就會自動交由init進(jìn)程來自動處理。
通過設(shè)定signal.signal(signal.SIGCHLD, signal.SIG_IGN)或join進(jìn)程可避免僵尸進(jìn)程的產(chǎn)生
def zombieProc():
print("zombie running")
time.sleep(5)
print("zombie exit")
if __name__ == '__main__':
signal.signal(signal.SIGCHLD, signal.SIG_IGN)
proc = multiprocessing.Process(target=zombieProc)
proc.start()
# proc.join()
time.sleep(30)Process類
Process([group [, target [, name [, args [, kwargs]]]]]),實(shí)例化得到的對象,表示一個子進(jìn)程任務(wù):
- group參數(shù)未使用,值始終為None;
- target表示調(diào)用對象,即子進(jìn)程要執(zhí)行的任務(wù);
- args表示調(diào)用對象的位置參數(shù)元組,args=(1, ‘test’, [‘one’]);
- kwargs表示調(diào)用對象的字典參數(shù),kwargs={‘name’:‘mike’,‘age’:18};
- name為子進(jìn)程的名稱;
Process類的屬性與方法:
- start():啟動進(jìn)程,并執(zhí)行其run方法;
- run():進(jìn)程啟動時運(yùn)行的方法,繼承Process類時必須要實(shí)現(xiàn)方法;
- terminate():強(qiáng)制終止進(jìn)程,不會進(jìn)行任何清理操作(若p創(chuàng)建了子進(jìn)程,則子進(jìn)程就成了僵尸進(jìn)程);如進(jìn)程還持有鎖等,那么也不會被釋放,進(jìn)而導(dǎo)致死鎖;
- is_alive():返回進(jìn)程是否在運(yùn)行狀態(tài);
- join([timeout]):等待進(jìn)程終止;
- daemon:默認(rèn)值為False,如果設(shè)為True,代表為守護(hù)進(jìn)程(當(dāng)父進(jìn)程終止時,隨之終止;并且不能創(chuàng)建自己的新進(jìn)程),必須在start()之前設(shè)置;
- name:進(jìn)程的名稱;
- pid/ident:進(jìn)程的pid;
- exitcode:進(jìn)程在運(yùn)行時為None、如果為–N,表示被信號N結(jié)束;
- authkey:進(jìn)程的身份驗(yàn)證碼(默認(rèn)是由os.urandom()隨機(jī)生成的32字符的字符串),在涉及網(wǎng)絡(luò)連接的底層進(jìn)程間通信時提供安全性;
也可通過os.getpid()獲取進(jìn)程的PID,os.getppid()獲取父進(jìn)程的PID。
函數(shù)方式
通過Process類直接運(yùn)行函數(shù):
def simpleRoutine(name, delay):
print(f"routine {name} starting...")
time.sleep(delay)
print(f"routine {name} finished")
if __name__ == '__main__':
thrOne = multiprocessing.Process(target=simpleRoutine, args=("First", 1))
thrTwo = multiprocessing.Process(target=simpleRoutine, args=("Two", 2))
thrOne.start()
thrTwo.start()
thrOne.join()
thrTwo.join()
繼承方式
通過繼承Process類,并實(shí)現(xiàn)run方法來啟動進(jìn)程:
class SimpleProcess(multiprocessing.Process):
def __init__(self, name, delay):
super().__init__()
self.name = name
self.delay = delay
def run(self):
print(f"Process {self.name} starting...")
time.sleep(self.delay)
print(f"Process {self.name} finished")
if __name__ == '__main__':
thrOne = SimpleProcess("First", 2)
thrTwo = SimpleProcess("Second", 1)
thrOne.start()
thrTwo.start()
thrOne.join()
thrTwo.join()
同步機(jī)制
進(jìn)程間同步與線程間同步類似(只是所有對象都在multiprocessing模塊中):
- Lock/Rlock: 通過acquire()和release()來獲取與釋放鎖;
- Event: 事件信號,通過set()和clear()來設(shè)定與清楚信號;通過wait()來等待信號;
- Condition: 條件變量;通過wait()用來等待條件,通過notify/notify_all()來通知等待此條件的進(jìn)程(等待與通知前,都需先持有鎖);
- Semaphore: 信號量;維護(hù)一個計(jì)數(shù)器;
- Barrier: 屏障;只有等待進(jìn)程數(shù)量達(dá)到要求數(shù)量,才會同時開始執(zhí)行屏障保護(hù)后的代碼。
屏障示例:
def waitBarrier(name, barr: multiprocessing.Barrier):
print(f"{name} waiting for open")
try:
barr.wait()
print(f"{name} running")
time.sleep(2)
except multiprocessing.BrokenBarrierError:
print(f"{name} exception")
print(f"{name} finished")
def openFun(): # 屏障滿足條件時,執(zhí)行一次
print("barrier opened")
if __name__ == '__main__':
signal = multiprocessing.Barrier(5, openFun)
for i in range(10):
multiprocessing.Process(target=waitBarrier, args=(i, signal)).start()
time.sleep(1)
當(dāng)?shù)?個進(jìn)程啟動時,前面5個進(jìn)程會同時開始執(zhí)行(openFun函數(shù)會執(zhí)行一次);當(dāng)?shù)?0個進(jìn)程啟動時,后面5個進(jìn)程會同時開始執(zhí)行一次(openFun函數(shù)又會執(zhí)行一次)。
狀態(tài)管理Managers
Managers提供了一種創(chuàng)建由多進(jìn)程(包括跨機(jī)器間進(jìn)程共享)共享的數(shù)據(jù)的方式:
multiprocessing.Manager()返回一個SyncManager對象;此對象對應(yīng)著一個管理者子進(jìn)程(manager process)以及代理(其他子進(jìn)程使用);- 它確保當(dāng)某一進(jìn)程修改了共享對象之后,其他進(jìn)程中的共享對象也會得到更新;
- 其支持的類型有:list、dict、Namespace、Lock、RLock、Semaphore、BoundedSemaphore、Condition、Event、Queue、Value和Array。
多進(jìn)程進(jìn)共享字典與列表(每個進(jìn)程中都能看到其他進(jìn)程修改過的內(nèi)容)
def worker(dictContext: dict, lstContext: list, name):
pid = os.getpid()
dictContext[name] = pid
lstContext.append(pid)
print(f"{name} worker: {lstContext}")
def managerContext():
mgr = multiprocessing.Manager()
multiprocessing.managers
dictContext = mgr.dict()
lstContext = mgr.list()
jobs = [multiprocessing.Process(target=worker, args=(dictContext, lstContext, i)) for i in range(10)]
for j in jobs:
j.start()
for j in jobs:
j.join()
print('Results:', dictContext)
到此這篇關(guān)于Python多進(jìn)程并發(fā)與同步機(jī)制超詳細(xì)講解的文章就介紹到這了,更多相關(guān)Python多進(jìn)程并發(fā)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
PyTorch中torch.manual_seed()的用法實(shí)例詳解
在Pytorch中可以通過相關(guān)隨機(jī)數(shù)來生成張量,并且可以指定生成隨機(jī)數(shù)的分布函數(shù)等,下面這篇文章主要給大家介紹了關(guān)于PyTorch中torch.manual_seed()用法的相關(guān)資料,需要的朋友可以參考下2022-06-06
python 與GO中操作slice,list的方式實(shí)例代碼
這篇文章主要介紹了python 與GO中操作slice,list的方式實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2017-03-03
超簡單的scrapy實(shí)現(xiàn)ip動態(tài)代理與更換ip的方法實(shí)現(xiàn)
這篇文章主要介紹了超簡單的scrapy實(shí)現(xiàn)ip動態(tài)代理與更換ip的方法實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03
python保留小數(shù)函數(shù)的幾種使用總結(jié)
本文主要介紹了python保留小數(shù)函數(shù)的幾種使用總結(jié),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-02-02
python通過yield實(shí)現(xiàn)數(shù)組全排列的方法
這篇文章主要介紹了python通過yield實(shí)現(xiàn)數(shù)組全排列的方法,實(shí)例分析了全排列的概念及yield實(shí)現(xiàn)的技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-03-03
Python?GUI和游戲開發(fā)從入門到實(shí)踐
GUI是圖形用戶界面的縮寫,圖形化的用戶界面對使用過計(jì)算機(jī)的人來說應(yīng)該都不陌生,下面這篇文章主要給大家介紹了關(guān)于Python圖形用戶界面與游戲開發(fā)的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-05-05
Python OpenCV圖像指定區(qū)域裁剪的實(shí)現(xiàn)
Python數(shù)據(jù)讀寫之Python讀寫CSV文件

