Python作為一門比較常見的編程語言,可以對其進行多線程的編程,包括利用Lock與RLock,實現(xiàn)多線程之間的相互同步,那么,實現(xiàn)這種原理的機制到底是什么樣子的呢?下面就給大家詳細解答下。
什么是鎖機制?
要回答這個問題,我們需要知道為什么需要使用鎖機制。前面我們談到一個進程內(nèi)的多個線程的某些資源是共享的,這也是線程的一大優(yōu)勢,但是也隨之帶來一個問題,即當兩個及兩個以上的線程同時訪問共享資源時,如果此時沒有預(yù)設(shè)對應(yīng)的同步機制,就可能帶來同一時刻多個線程同時訪問同一個共享資源,即出現(xiàn)競態(tài),多數(shù)情況下我們是不希望出現(xiàn)這樣的情況的,那么怎么避免呢?
Lock()管理線程
先看一段代碼:
import threading import time resource=0 count=1000000 resource_lock=threading.Lock() def increment(): global resource for i in range(count): resource+=1 def decerment(): global resource for i in range(count): resource-=1 increment_thread=threading.Thread(target=increment) decerment_thread=threading.Thread(target=decerment) increment_thread.start() decerment_thread.start() increment_thread.join() decerment_thread.join() print(resource)
運行截圖如下:
運行結(jié)果
當我們多次運行時,可以看到最終的結(jié)果都幾乎不等于我們期待的值即resource初始值0。
為什么呢?原因就是因為+=和-=并不是原子操作。
可以使用dis模塊查看字節(jié)碼:
import dis def add(total): total+=1 def desc(total): total-=1 total=0 print(dis.dis(add)) print(dis.dis(desc)) #運行結(jié)果: #3 0 LOAD_FAST 0(total) #3 LOAD_CONST 1(1) #6 INPLACE_ADD #7 STORE_FAST 0(total) #10 LOAD_CONST 0(None) #13 RETURN_VALUE #None #5 0 LOAD_FAST 0(total) #3 LOAD_CONST 1(1) #6 INPLACE_SUBTRACT #7 STORE_FAST 0(total) #10 LOAD_CONST 0(None) #13 RETURN_VALUE #None
那么如何保證初始值為0呢?我們可以利用Lock(),代碼如下:
import threading import time resource=0 count=1000000 resource_lock=threading.Lock() def increment(): global resource for i in range(count): resource_lock.acquire() resource+=1 resource_lock.release() def decerment(): global resource for i in range(count): resource_lock.acquire() resource-=1 resource_lock.release() increment_thread=threading.Thread(target=increment) decerment_thread=threading.Thread(target=decerment) increment_thread.start() decerment_thread.start() increment_thread.join() decerment_thread.join() print(resource)
運行截圖如下:
運行結(jié)果
從運行結(jié)果可以看到,不論我們運行多少次改代碼,其resource的值都為初始值0,這就是Lock()的功勞,即它可以將某一時刻的訪問限定在單個線程或者單個類型的線程上,在訪問鎖定的共享資源時,必須要現(xiàn)獲取對應(yīng)的鎖才能訪問,即要等待其他線程釋放資源,即resource_lock.release()當然為了防止我們對某個資源鎖定后,忘記釋放鎖,導(dǎo)致死鎖,我們可以利用上下文管理器管理鎖實現(xiàn)同樣的效果:
import threading import time resource=0 count=1000000 resource_lock=threading.Lock() def increment(): global resource for i in range(count): with resource_lock: resource+=1 def decerment(): global resource for i in range(count): with resource_lock: resource-=1 increment_thread=threading.Thread(target=increment) decerment_thread=threading.Thread(target=decerment) increment_thread.start() decerment_thread.start()
RLock()與Lock()的區(qū)別
我們需要知道Lock()作為一個基本的鎖對象,一次只能一個鎖定,其余鎖請求,需等待鎖釋放后才能獲取,否則會發(fā)生死鎖:
import threading resource.lock=threading.lock() resource=0 resource.lock.acquire() resource.lock.acquire() resource+=1 resource.lock.release() resource.lock.release()
為解決同一線程中不能多次請求同一資源的問題,python提供了“可重入鎖”:threading.RLock,RLock內(nèi)部維護著一個Lock和一個counter變量,counter記錄了acquire的次數(shù),從而使得資源可以被多次acquire。
直到一個線程所有的acquire都被release,其他的線程才能獲得資源。用法和threading.Lock類相同,即比如遞歸鎖的使用:
import threading lock=threading.RLock() def dosomething(lock): lock.acquire() #do something lock.release() lock.acquire() dosomething(lock) lock.release()
綜上所述,這篇文章就給大家介紹到這里了,希望可以給大家?guī)韼椭?/p>
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/128429.html
摘要:文章結(jié)構(gòu)來自七周七并發(fā)模型互斥和內(nèi)存模型創(chuàng)建線程這段代碼創(chuàng)建并啟動了一個實例,首先從開始,函數(shù)的余下部分一起并發(fā)執(zhí)行。在鎖定狀態(tài)下,某些線程擁有鎖在非鎖定狀態(tài)下,沒有線程擁有它。 并發(fā)&并行 并發(fā)程序含有多個邏輯上的獨立執(zhí)行塊,他們可以獨立的并行執(zhí)行,也可以串行執(zhí)行。并行程序解決問題的速度比串行程序快的多,因為其可以同時執(zhí)行整個任務(wù)的多個部分。并行程序可能有多個獨立執(zhí)行塊,也可能只有一...
摘要:如果某線程并未使用很多操作,它會在自己的時間片內(nèi)一直占用處理器和。在中使用線程在和等大多數(shù)類系統(tǒng)上運行時,支持多線程編程。守護線程另一個避免使用模塊的原因是,它不支持守護線程。 這一篇是Python并發(fā)的第四篇,主要介紹進程和線程的定義,Python線程和全局解釋器鎖以及Python如何使用thread模塊處理并發(fā) 引言&動機 考慮一下這個場景,我們有10000條數(shù)據(jù)需要處理,處理每條...
摘要:某進程內(nèi)的線程在其它進程不可見。線程的實體包括程序數(shù)據(jù)和。包括以下信息線程狀態(tài)。當線程不運行時,被保存的現(xiàn)場資源。用戶級線程執(zhí)行系統(tǒng)調(diào)用指令時將導(dǎo)致其所屬進程被中斷,而內(nèi)核支持線程執(zhí)行系統(tǒng)調(diào)用指令時,只導(dǎo)致該線程被中斷。線程能夠利用的表空 操作系統(tǒng)線程理論 線程概念的引入背景 進程之前我們已經(jīng)了解了操作系統(tǒng)中進程的概念,程序并不能單獨運行,只有將程序裝載到內(nèi)存中,系統(tǒng)為它分配資源才能運...
摘要:本文最先發(fā)布在博客這篇文章將講解并發(fā)編程的基本操作。并發(fā)是指能夠多任務(wù)處理,并行則是是能夠同時多任務(wù)處理。雖然自帶了很好的類庫支持多線程進程編程,但眾所周知,因為的存在,很難做好真正的并行。 本文最先發(fā)布在博客:https://blog.ihypo.net/151628... 這篇文章將講解 Python 并發(fā)編程的基本操作。并發(fā)和并行是對孿生兄弟,概念經(jīng)?;煜?。并發(fā)是指能夠多任務(wù)處...
閱讀 923·2023-01-14 11:38
閱讀 895·2023-01-14 11:04
閱讀 756·2023-01-14 10:48
閱讀 2055·2023-01-14 10:34
閱讀 961·2023-01-14 10:24
閱讀 840·2023-01-14 10:18
閱讀 510·2023-01-14 10:09
閱讀 588·2023-01-14 10:02