欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時(shí)間:2020年09月28日 08:38:07   作者:編程的朝圣之路  
這篇文章主要介紹了互斥鎖解決 Python 中多線程共享全局變量的問(wèn)題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下

一、同步概念

同步就是協(xié)同步調(diào),按預(yù)定的先后次序進(jìn)行運(yùn)行。如:你說(shuō)完,我再說(shuō)。

"同"字從字面上容易理解為一起動(dòng)作。

其實(shí)不是,在這里,"同"字應(yīng)是指協(xié)同、協(xié)助、互相配合。

線程同步,可理解為線程A和B一塊配合,A執(zhí)行到一定程度時(shí)要依靠B的某個(gè)結(jié)果,于是停下來(lái),示意B運(yùn)行;B執(zhí)行,再將結(jié)果給A;A再繼續(xù)操作。

之前我們遇到過(guò),如果多個(gè)線程共同對(duì)某個(gè)數(shù)據(jù)修改,則可能出現(xiàn)不可預(yù)料的結(jié)果,為了保證數(shù)據(jù)的正確性,需要對(duì)多個(gè)線程進(jìn)行同步。

解決線程同時(shí)修改全局變量的方式

我們先把上次那個(gè)問(wèn)題再看下。

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("---線程創(chuàng)建之前g_num is %d---" % g_num)

t1 = threading.Thread(target=work1, args=(1000000,))
t1.start()

t2 = threading.Thread(target=work2, args=(1000000,))
t2.start()

# 確保子線程都運(yùn)行結(jié)束
while len(threading.enumerate()) != 1:
 time.sleep(1)

print("2個(gè)線程對(duì)同一個(gè)全局變量操作之后的最終結(jié)果是:%s" % g_num)

運(yùn)行結(jié)果:

---線程創(chuàng)建之前g_num is 0---
----in work2, g_num is 1048576---
----in work1, g_num is 1155200---
2個(gè)線程對(duì)同一個(gè)全局變量操作之后的最終結(jié)果是:1155200

對(duì)于這個(gè)計(jì)算錯(cuò)誤的問(wèn)題,可以通過(guò)線程同步來(lái)進(jìn)行解決。

思路,如下:

系統(tǒng)調(diào)用 t1,然后獲取到 g_num 的值為0,此時(shí)上一把鎖,即不允許其他線程操作 g_num。

t1 對(duì) g_num 的值進(jìn)行+1。

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

同理其他線程在對(duì) g_num 進(jìn)行修改時(shí),都要先上鎖,處理完后再解鎖,在上鎖的整個(gè)過(guò)程中不允許其他線程訪問(wèn),就保證了數(shù)據(jù)的正確性。

思路基本是這個(gè)樣子,那代碼怎么來(lái)實(shí)現(xiàn)呢?

二、互斥鎖解決資源競(jìng)爭(zhēng)的問(wèn)題

當(dāng)多個(gè)線程幾乎同時(shí)修改某一個(gè)共享數(shù)據(jù)的時(shí)候,需要進(jìn)行同步控制。

線程同步能夠保證多個(gè)線程安全訪問(wèn)競(jìng)爭(zhēng)資源,最簡(jiǎn)單的同步機(jī)制就是引入互斥鎖。

互斥鎖為資源引入一個(gè)狀態(tài):鎖定/非鎖定。

某個(gè)線程要更改共享數(shù)據(jù)時(shí),先將其鎖定,此時(shí)資源的狀態(tài)為“鎖定”,其他線程不能更改;直到該線程釋放資源,將資源的狀態(tài)變成“非鎖定”,其他的線程才能再次鎖定該資源。

互斥鎖保證了每次只有一個(gè)線程進(jìn)行寫(xiě)入操作,從而保證了多線程情況下數(shù)據(jù)的正確性。

threading 模塊中定義了 Lock 類(lèi),可以方便的處理鎖定:

# 創(chuàng)建鎖
mutex = threading.Lock()

# 鎖定
mutex.acquire()

# 釋放
mutex.release()

注意:

如果這個(gè)鎖之前是沒(méi)有上鎖的,那么 acquire 不會(huì)堵塞。

如果在調(diào)用 acquire 對(duì)這個(gè)鎖上鎖之前,它已經(jīng)被其他線程上了鎖,那么此時(shí) acquire 會(huì)堵塞,直到這個(gè)鎖被解鎖為止。

示例:

使用互斥鎖完成2個(gè)線程對(duì)同一個(gè)全局變量各加100萬(wàn)次的操作。

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)

# 創(chuàng)建一個(gè)互斥鎖
# 默認(rèn)是未上鎖的狀態(tài)
mutex = threading.Lock()

# 創(chuàng)建2個(gè)線程,讓他們各自對(duì)g_num加1000000次
p1 = threading.Thread(target=test1, args=(1000000,))
p1.start()

p2 = threading.Thread(target=test2, args=(1000000,))
p2.start()

# 等待計(jì)算完成
while len(threading.enumerate()) != 1:
 time.sleep(1)

print("2個(gè)線程對(duì)同一個(gè)全局變量操作之后的最終結(jié)果是:%s" % g_num)

運(yùn)行結(jié)果:

---test1---g_num=1989108
---test2---g_num=2000000
2個(gè)線程對(duì)同一個(gè)全局變量操作之后的最終結(jié)果是:2000000

可以看到最后的結(jié)果,加入互斥鎖后,其結(jié)果與預(yù)期相符。

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

上鎖解鎖過(guò)程

當(dāng)一個(gè)線程調(diào)用鎖的 acquire() 方法獲得鎖時(shí),鎖就進(jìn)入“l(fā)ocked”狀態(tài)。

每次只有一個(gè)線程可以獲得鎖。

如果此時(shí)另一個(gè)線程試圖獲得這個(gè)鎖,該線程就會(huì)變?yōu)椤癰locked”狀態(tài),稱(chēng)為“阻塞”,直到擁有鎖的線程調(diào)用鎖的 release() 方法釋放鎖之后,鎖進(jìn)入“unlocked”狀態(tài)。

線程調(diào)度程序從處于同步阻塞狀態(tài)的線程中選擇一個(gè)來(lái)獲得鎖,并使得該線程進(jìn)入運(yùn)行(running)狀態(tài)。

總結(jié)

鎖的好處:

確保了某段關(guān)鍵代碼只能由一個(gè)線程從頭到尾完整地執(zhí)行。

鎖的壞處:

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

由于可以存在多個(gè)鎖,不同的線程持有不同的鎖,并試圖獲取對(duì)方持有的鎖時(shí),可能會(huì)造成死鎖。

到此這篇關(guān)于互斥鎖解決 Python 中多線程共享全局變量的問(wèn)題的文章就介紹到這了,更多相關(guān)Python 多線程共享全局變量?jī)?nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Python實(shí)現(xiàn)基于HTTP文件傳輸實(shí)例

    Python實(shí)現(xiàn)基于HTTP文件傳輸實(shí)例

    這篇文章主要介紹了Python實(shí)現(xiàn)基于HTTP文件傳輸?shù)姆椒?以實(shí)例形式詳細(xì)講述了server端與client端的實(shí)現(xiàn)代碼,非常具有實(shí)用價(jià)值,需要的朋友可以參考下
    2014-11-11
  • 淺談python的elementtree模塊處理中文注意事項(xiàng)

    淺談python的elementtree模塊處理中文注意事項(xiàng)

    這篇文章主要介紹了淺談python的elementtree模塊處理中文注意事項(xiàng),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-03-03
  • Python學(xué)習(xí)之加密模塊使用詳解

    Python學(xué)習(xí)之加密模塊使用詳解

    加密模塊在工作中被廣泛應(yīng)用。比如數(shù)據(jù)的傳入 不希望被捕獲,通過(guò)把數(shù)據(jù)加密,這樣即使被捕獲也無(wú)法獲取到數(shù)據(jù)的真實(shí)信息。本文將學(xué)習(xí)一下Python中的加密模塊的使用 方法,需要的可以參考一下
    2022-03-03
  • Python腳本實(shí)現(xiàn)蝦米網(wǎng)簽到功能

    Python腳本實(shí)現(xiàn)蝦米網(wǎng)簽到功能

    這篇文章主要介紹了Python腳本實(shí)現(xiàn)蝦米網(wǎng)簽到功能的方法,涉及Python調(diào)用URL模塊實(shí)現(xiàn)數(shù)據(jù)傳輸與處理的相關(guān)技巧,需要的朋友可以參考下
    2016-04-04
  • pytorch人工智能之torch.gather算子用法示例

    pytorch人工智能之torch.gather算子用法示例

    這篇文章主要介紹了pytorch人工智能之torch.gather算子用法示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-09-09
  • Python?中?Kwargs?解析的最佳實(shí)踐教程

    Python?中?Kwargs?解析的最佳實(shí)踐教程

    這篇文章主要介紹了Python中Kwargs解析的最佳實(shí)踐,使用?kwargs,我們可以編寫(xiě)帶有任意數(shù)量關(guān)鍵字參數(shù)的函數(shù),當(dāng)我們想為函數(shù)提供靈活的接口時(shí),這會(huì)很有用,需要的朋友可以參考下
    2023-06-06
  • python學(xué)習(xí)開(kāi)發(fā)mock接口

    python學(xué)習(xí)開(kāi)發(fā)mock接口

    這篇文章主要為大家詳細(xì)介紹了python學(xué)習(xí)開(kāi)發(fā)mock接口的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-04-04
  • Django零基礎(chǔ)入門(mén)之自定義過(guò)濾器及模板中的使用

    Django零基礎(chǔ)入門(mén)之自定義過(guò)濾器及模板中的使用

    這篇文章主要介紹了Django零基礎(chǔ)入門(mén)之自定義過(guò)濾器及模板中的使用,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-09-09
  • Python面向?qū)ο笾^承原理與用法案例分析

    Python面向?qū)ο笾^承原理與用法案例分析

    這篇文章主要介紹了Python面向?qū)ο笾^承原理與用法,結(jié)合具體案例形式分析了Python面向?qū)ο蟪绦蛟O(shè)計(jì)中繼承的原理、使用方法及相關(guān)操作注意事項(xiàng),需要的朋友可以參考下
    2019-12-12
  • Python基礎(chǔ)之模塊相關(guān)知識(shí)總結(jié)

    Python基礎(chǔ)之模塊相關(guān)知識(shí)總結(jié)

    今天帶大家復(fù)習(xí)Python基礎(chǔ)知識(shí),文中對(duì)模塊相關(guān)知識(shí)介紹的非常詳細(xì),對(duì)正在學(xué)習(xí)python基礎(chǔ)的小伙伴們很有幫助,需要的朋友可以參考下
    2021-05-05

最新評(píng)論