Python threading模塊中l(wèi)ock與Rlock的使用詳細(xì)講解
前言
在使用多線程的應(yīng)用下,如何保證線程安全,以及線程之間的同步,或者訪問共享變量等問題是十分棘手的問題,也是使用多線程下面臨的問題,如果處理不好,會帶來較嚴(yán)重的后果,使用python多線程中提供Lock 、Rlock 、Semaphore 、Event 、Condition 用來保證線程之間的同步,后者保證訪問共享變量的互斥問題。
- Lock & RLock:互斥鎖,用來保證多線程訪問共享變量的問題
- Semaphore對象:Lock互斥鎖的加強版,可以被多個線程同時擁有,而Lock只能被某一個線程同時擁有。
- Event對象:它是線程間通信的方式,相當(dāng)于信號,一個線程可以給另外一個線程發(fā)送信號后讓其執(zhí)行操作。
- Condition對象:其可以在某些事件觸發(fā)或者達(dá)到特定的條件后才處理數(shù)據(jù)
1、Lock(互斥鎖)
請求鎖定 — 進入鎖定池等待 — — 獲取鎖 — 已鎖定— — 釋放鎖
Lock(指令鎖)是可用的最低級的同步指令。Lock處于鎖定狀態(tài)時,不被特定的線程擁有。Lock包含兩種狀態(tài)——鎖定和非鎖定,以及兩個基本的方法。
可以認(rèn)為Lock有一個鎖定池,當(dāng)線程請求鎖定時,將線程至于池中,直到獲得鎖定后出池。池中的線程處于狀態(tài)圖中的同步阻塞狀態(tài)。
構(gòu)造方法:mylock = Threading.Lock( )
實例方法:
- acquire([timeout]): 使線程進入同步阻塞狀態(tài),嘗試獲得鎖定。
- release(): 釋放鎖。使用前線程必須已獲得鎖定,否則將拋出異常。
實例一(未使用鎖):
import threading
import time
num = 0
def show(arg):
global num
time.sleep(1)
num +=1
print('bb :{}'.format(num))
for i in range(5):
t = threading.Thread(target=show, args=(i,)) # 注意傳入?yún)?shù)后一定要有【,】逗號
t.start()
print('main thread stop')
main thread stop
bb :1
bb :2
bb :3bb :4
bb :5
實例二(使用鎖)
import threading
import time
num = 0
lock = threading.RLock()
# 調(diào)用acquire([timeout])時,線程將一直阻塞,
# 直到獲得鎖定或者直到timeout秒后(timeout參數(shù)可選)。
# 返回是否獲得鎖。
def Func():
lock.acquire()
global num
num += 1
time.sleep(1)
print(num)
lock.release()
for i in range(10):
t = threading.Thread(target=Func)
t.start()
1
2
3
4
5
6
7
8
9
10
#可以看出,全局變量在在每次被調(diào)用時都要獲得鎖,才能操作,因此保證了共享數(shù)據(jù)的安全性
對于Lock對象而言,如果一個線程連續(xù)兩次release,使得線程死鎖。所以Lock不常用,一般采用Rlock進行線程鎖的設(shè)定。
import threading
import time
class MyThread(threading.Thread):
def run(self):
global num
time.sleep(1)
if lock.acquire(1):
num = num+1
msg = self.name+' set num to '+str(num)
print(msg)
lock.acquire()
lock.release()
lock.release()
num = 0
lock = threading.Lock()
def test():
for i in range(5):
t = MyThread()
t.start()
if __name__ == '__main__':
test()
Thread-12 set num to 12、RLock(可重入鎖)
RLock(可重入鎖)是一個可以被同一個線程請求多次的同步指令。RLock使用了“擁有的線程”和“遞歸等級”的概念,處于鎖定狀態(tài)時,RLock被某個線程擁有。擁有RLock的線程可以再次調(diào)用acquire(),釋放鎖時需要調(diào)用release()相同次數(shù)??梢哉J(rèn)為RLock包含一個鎖定池和一個初始值為0的計數(shù)器,每次成功調(diào)用 acquire()/release(),計數(shù)器將+1/-1,為0時鎖處于未鎖定狀態(tài)。
- 構(gòu)造方法:mylock = Threading.RLock()
- 實例方法:acquire([timeout])/release(): 跟Lock差不多。
實例解決死鎖,調(diào)用相同次數(shù)的acquire和release,保證成對出現(xiàn)
''' 尋找有志同道合的小伙伴,互幫互助,群里還有不錯的視頻學(xué)習(xí)教程和PDF電子書! ''' import threading rLock = threading.RLock() #RLock對象 rLock.acquire() rLock.acquire() #在同一線程內(nèi),程序不會堵塞。 rLock.release() rLock.release() print(rLock.acquire())
詳細(xì)實例:
import threading
mylock = threading.RLock()
num = 0
class WorkThread(threading.Thread):
def __init__(self, name):
threading.Thread.__init__(self)
self.t_name = name
def run(self):
global num
while True:
mylock.acquire()
print('\n%s locked, number: %d' % (self.t_name, num))
if num >= 2:
mylock.release()
print('\n%s released, number: %d' % (self.t_name, num))
break
num += 1
print('\n%s released, number: %d' % (self.t_name, num))
mylock.release()
def test():
thread1 = WorkThread('A-Worker')
thread2 = WorkThread('B-Worker')
thread1.start()
thread2.start()
if __name__ == '__main__':
test() A-Worker locked, number: 0
A-Worker released, number: 1
A-Worker locked, number: 1
A-Worker released, number: 2
A-Worker locked, number: 2
A-Worker released, number: 2
B-Worker locked, number: 2
B-Worker released, number: 2
到此這篇關(guān)于Python threading模塊中l(wèi)ock與Rlock的使用詳細(xì)講解的文章就介紹到這了,更多相關(guān)Python threading內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python庫Cerberus數(shù)據(jù)驗證如何保證系統(tǒng)安全性正確性實例分析
本文帶大家探索?Cerberus?庫,看看它是如何幫助我們在開發(fā)過程中保障數(shù)據(jù)的安全性和完整性的,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2024-01-01
如何利用python在剪貼板上讀取/寫入數(shù)據(jù)
說起處理數(shù)據(jù)就離不開導(dǎo)入導(dǎo)出,而我們使用Pandas時候最常用的就是read_excel、read_csv了,下面這篇文章主要給大家介紹了關(guān)于如何利用python在剪貼板上讀取/寫入數(shù)據(jù)的相關(guān)資料,需要的朋友可以參考下2022-07-07
python庫patchworklib多圖表整合用法示例探究
這篇文章主要介紹了python庫patchworklib對齊matplotlib圖表,將多個圖表的整合為單一圖表用法示例探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2024-01-01
Python進階學(xué)習(xí)之帶你探尋Python類的鼻祖-元類
這篇文章主要介紹了Python進階學(xué)習(xí)之帶你探尋Python類的鼻祖-元類,文中有非常詳細(xì)的解釋,對正在學(xué)習(xí)python的小伙伴們有很好的幫助,需要的朋友可以參考下2021-05-05
Python機器學(xué)習(xí)之實現(xiàn)模型持久化與加載
在實際的機器學(xué)習(xí)項目中,我們通常需要將訓(xùn)練好的模型保存到磁盤,本文我們會介紹如何在Python中使用pickle和joblib庫將訓(xùn)練好的模型持久化到磁盤,需要的可以參考一下2023-05-05
pandas?dataframe獲取所有行名稱與列名稱方法示例
這篇文章主要給大家介紹了關(guān)于pandas?dataframe獲取所有行名稱與列名稱的相關(guān)資料,Pandas是Python中用于數(shù)據(jù)分析的非常重要的庫,它提供了多種方法來獲取列名,需要的朋友可以參考下2023-09-09
python實現(xiàn)坦克大戰(zhàn)游戲 附詳細(xì)注釋
這篇文章主要為大家詳細(xì)介紹了python實現(xiàn)坦克大戰(zhàn)游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-05-05
python實現(xiàn)從文件中讀取數(shù)據(jù)并繪制成 x y 軸圖形的方法
今天小編就為大家分享一篇python實現(xiàn)從文件中讀取數(shù)據(jù)并繪制成 x y 軸圖形的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-10-10

