一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

腳本之家,腳本語言編程技術及教程分享平臺!
分類導航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服務器之家 - 腳本之家 - Python - 互斥鎖解決 Python 中多線程共享全局變量的問題(推薦)

互斥鎖解決 Python 中多線程共享全局變量的問題(推薦)

2020-09-28 23:55編程的朝圣之路 Python

這篇文章主要介紹了互斥鎖解決 Python 中多線程共享全局變量的問題,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下

一、同步概念

同步就是協同步調,按預定的先后次序進行運行。如:你說完,我再說。

"同"字從字面上容易理解為一起動作。

其實不是,在這里,"同"字應是指協同、協助、互相配合。

線程同步,可理解為線程A和B一塊配合,A執行到一定程度時要依靠B的某個結果,于是停下來,示意B運行;B執行,再將結果給A;A再繼續操作。

之前我們遇到過,如果多個線程共同對某個數據修改,則可能出現不可預料的結果,為了保證數據的正確性,需要對多個線程進行同步。

解決線程同時修改全局變量的方式

我們先把上次那個問題再看下。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
import threading
import time
 
g_num = 0
 
def work1(num):
  global g_num
  for i in range(num):
    g_num += 1
  print("----in work1, g_num is %d---" % g_num)
 
def work2(num):
  global g_num
  for i in range(num):
    g_num += 1
  print("----in work2, g_num is %d---" % g_num)
 
print("---線程創建之前g_num is %d---" % g_num)
 
t1 = threading.Thread(target=work1, args=(1000000,))
t1.start()
 
t2 = threading.Thread(target=work2, args=(1000000,))
t2.start()
 
# 確保子線程都運行結束
while len(threading.enumerate()) != 1:
  time.sleep(1)
 
print("2個線程對同一個全局變量操作之后的最終結果是:%s" % g_num)

運行結果:

---線程創建之前g_num is 0---
----in work2, g_num is 1048576---
----in work1, g_num is 1155200---
2個線程對同一個全局變量操作之后的最終結果是:1155200

對于這個計算錯誤的問題,可以通過線程同步來進行解決。

思路,如下:

系統調用 t1,然后獲取到 g_num 的值為0,此時上一把鎖,即不允許其他線程操作 g_num。

t1 對 g_num 的值進行+1。

t1 解鎖,此時 g_num 的值為1,其他的線程就可以使用 g_num 了,而且 g_num 的值不是0而是1。

同理其他線程在對 g_num 進行修改時,都要先上鎖,處理完后再解鎖,在上鎖的整個過程中不允許其他線程訪問,就保證了數據的正確性。

思路基本是這個樣子,那代碼怎么來實現呢?

二、互斥鎖解決資源競爭的問題

當多個線程幾乎同時修改某一個共享數據的時候,需要進行同步控制。

線程同步能夠保證多個線程安全訪問競爭資源,最簡單的同步機制就是引入互斥鎖。

互斥鎖為資源引入一個狀態:鎖定/非鎖定。

某個線程要更改共享數據時,先將其鎖定,此時資源的狀態為“鎖定”,其他線程不能更改;直到該線程釋放資源,將資源的狀態變成“非鎖定”,其他的線程才能再次鎖定該資源。

互斥鎖保證了每次只有一個線程進行寫入操作,從而保證了多線程情況下數據的正確性。

互斥鎖解決 Python 中多線程共享全局變量的問題(推薦)

threading 模塊中定義了 Lock 類,可以方便的處理鎖定:

?
1
2
3
4
5
6
7
8
# 創建鎖
mutex = threading.Lock()
 
# 鎖定
mutex.acquire()
 
# 釋放
mutex.release()

注意:

如果這個鎖之前是沒有上鎖的,那么 acquire 不會堵塞。

如果在調用 acquire 對這個鎖上鎖之前,它已經被其他線程上了鎖,那么此時 acquire 會堵塞,直到這個鎖被解鎖為止。

示例:

使用互斥鎖完成2個線程對同一個全局變量各加100萬次的操作。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
import threading
import time
 
g_num = 0
 
def test1(num):
  global g_num
  for i in range(num):
    mutex.acquire() # 上鎖
    g_num += 1
    mutex.release() # 解鎖
 
  print("---test1---g_num=%d" % g_num)
 
def test2(num):
  global g_num
  for i in range(num):
    mutex.acquire() # 上鎖
    g_num += 1
    mutex.release() # 解鎖
 
  print("---test2---g_num=%d" % g_num)
 
# 創建一個互斥鎖
# 默認是未上鎖的狀態
mutex = threading.Lock()
 
# 創建2個線程,讓他們各自對g_num加1000000次
p1 = threading.Thread(target=test1, args=(1000000,))
p1.start()
 
p2 = threading.Thread(target=test2, args=(1000000,))
p2.start()
 
# 等待計算完成
while len(threading.enumerate()) != 1:
  time.sleep(1)
 
print("2個線程對同一個全局變量操作之后的最終結果是:%s" % g_num)

運行結果:

---test1---g_num=1989108
---test2---g_num=2000000
2個線程對同一個全局變量操作之后的最終結果是:2000000

可以看到最后的結果,加入互斥鎖后,其結果與預期相符。

記住,上鎖的代碼范圍要越小越好。在業務邏輯正確的前提下,能鎖一行代碼,就不要鎖兩行。

上鎖解鎖過程

當一個線程調用鎖的 acquire() 方法獲得鎖時,鎖就進入“locked”狀態。

每次只有一個線程可以獲得鎖。

如果此時另一個線程試圖獲得這個鎖,該線程就會變為“blocked”狀態,稱為“阻塞”,直到擁有鎖的線程調用鎖的 release() 方法釋放鎖之后,鎖進入“unlocked”狀態。

線程調度程序從處于同步阻塞狀態的線程中選擇一個來獲得鎖,并使得該線程進入運行(running)狀態。

總結

鎖的好處:

確保了某段關鍵代碼只能由一個線程從頭到尾完整地執行。

鎖的壞處:

阻止了多線程并發執行,包含鎖的某段代碼實際上只能以單線程模式執行,效率就大大地下降了。

由于可以存在多個鎖,不同的線程持有不同的鎖,并試圖獲取對方持有的鎖時,可能會造成死鎖。

到此這篇關于互斥鎖解決 Python 中多線程共享全局變量的問題的文章就介紹到這了,更多相關Python 多線程共享全局變量內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家!

原文鏈接:https://www.cnblogs.com/studyming/archive/2020/09/28/13742958.html

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 8x8x极品国产在线 | 国产第一页无线好源 | 日本精品一二三区 | 800精品国产导航 | 国产精品亚洲专区在线播放 | 麻豆视频网 | 欧美s级人做人爱c视频 | 国产大片线上免费观看 | 久久se精品一区二区国产 | 人人澡 人人澡碰人人看软件 | 美女脱了内裤打开腿让人羞羞软件 | 婷婷综合亚洲 | sese在线播放 | 东北美女野外bbwbbw免费 | aaa级黄色片| 亚洲国产99999在线精品一区 | 亚洲精品成人AV在线观看爽翻 | 99精品国产成人a∨免费看 | 亚洲天堂精品在线观看 | 视频一区国产精戏刘婷30 | piss美女厕所小便 | 久久高清一级毛片 | 九九九精品视频 | 4hc44四虎www在线影院男同 | 亚洲www在线 | 我和子伦系列小说 | 四虎成人免费观看在线网址 | 亚洲高清国产拍精品影院 | 欧美日韩1区2区 | 天堂69亚洲精品中文字幕 | 出差上的少妇20p | 被夫上司强迫中文 | 青草国内精品视频在线观看 | 欧美一级专区免费大片俄罗斯 | 丝袜兔女郎被啪在线观看91 | 欧美最猛性xxxxx短视频 | 日韩aaa | 好男人影视社区www在线观看 | 免费日批软件 | 无码爽死成人777在线观看网站 | 亚洲精品专区 |