網站首頁 編程語言 正文
前言
在使用多線程的應用下,如何保證線程安全,以及線程之間的同步,或者訪問共享變量等問題是十分棘手的問題,也是使用多線程下面臨的問題,如果處理不好,會帶來較嚴重的后果,使用python多線程中提供Lock 、Rlock 、Semaphore 、Event 、Condition 用來保證線程之間的同步,后者保證訪問共享變量的互斥問題。
- Lock & RLock:互斥鎖,用來保證多線程訪問共享變量的問題
- Semaphore對象:Lock互斥鎖的加強版,可以被多個線程同時擁有,而Lock只能被某一個線程同時擁有。
- Event對象:它是線程間通信的方式,相當于信號,一個線程可以給另外一個線程發送信號后讓其執行操作。
- Condition對象:其可以在某些事件觸發或者達到特定的條件后才處理數據
1、Lock(互斥鎖)
請求鎖定 — 進入鎖定池等待 — — 獲取鎖 — 已鎖定— — 釋放鎖
Lock(指令鎖)是可用的最低級的同步指令。Lock處于鎖定狀態時,不被特定的線程擁有。Lock包含兩種狀態——鎖定和非鎖定,以及兩個基本的方法。
可以認為Lock有一個鎖定池,當線程請求鎖定時,將線程至于池中,直到獲得鎖定后出池。池中的線程處于狀態圖中的同步阻塞狀態。
構造方法:mylock = Threading.Lock( )
實例方法:
- acquire([timeout]): 使線程進入同步阻塞狀態,嘗試獲得鎖定。
- 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,)) # 注意傳入參數后一定要有【,】逗號
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()
# 調用acquire([timeout])時,線程將一直阻塞,
# 直到獲得鎖定或者直到timeout秒后(timeout參數可選)。
# 返回是否獲得鎖。
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
#可以看出,全局變量在在每次被調用時都要獲得鎖,才能操作,因此保證了共享數據的安全性
對于Lock對象而言,如果一個線程連續兩次release,使得線程死鎖。所以Lock不常用,一般采用Rlock進行線程鎖的設定。
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 1
2、RLock(可重入鎖)
RLock(可重入鎖)是一個可以被同一個線程請求多次的同步指令。RLock使用了“擁有的線程”和“遞歸等級”的概念,處于鎖定狀態時,RLock被某個線程擁有。擁有RLock的線程可以再次調用acquire(),釋放鎖時需要調用release()相同次數。可以認為RLock包含一個鎖定池和一個初始值為0的計數器,每次成功調用 acquire()/release(),計數器將+1/-1,為0時鎖處于未鎖定狀態。
- 構造方法:mylock = Threading.RLock()
- 實例方法:acquire([timeout])/release(): 跟Lock差不多。
實例解決死鎖,調用相同次數的acquire和release,保證成對出現
'''
尋找有志同道合的小伙伴,互幫互助,群里還有不錯的視頻學習教程和PDF電子書!
'''
import threading
rLock = threading.RLock() #RLock對象
rLock.acquire()
rLock.acquire() #在同一線程內,程序不會堵塞。
rLock.release()
rLock.release()
print(rLock.acquire())
詳細實例:
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
原文鏈接:https://blog.csdn.net/qdPython/article/details/127073636
相關推薦
- 2022-03-22 .NET?6開發TodoList實現請求日志組件HttpLogging_實用技巧
- 2021-12-04 Flutter實現頁面路由及404路由攔截_Android
- 2022-05-11 在 Markdown 中,如何在反引號對語句中使用反引號
- 2022-05-18 python?安全地刪除列表元素的方法_python
- 2023-03-04 React深入分析更新的創建源碼_React
- 2023-01-26 Android入門之使用SQLite內嵌式數據庫詳解_Android
- 2023-08-28 axios的使用和接口請求統一封裝處理
- 2022-07-21 常見CSS樣式
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支