日韩欧乱色一区二区三区在线_久久―日本道色综合久久_欧美日本一区二区_网曝91综合精品门事件在线

千鋒教育-做有情懷、有良心、有品質(zhì)的職業(yè)教育機(jī)構(gòu)

手機(jī)站
千鋒教育

千鋒學(xué)習(xí)站 | 隨時(shí)隨地免費(fèi)學(xué)

千鋒教育

掃一掃進(jìn)入千鋒手機(jī)站

領(lǐng)取全套視頻
千鋒教育

關(guān)注千鋒學(xué)習(xí)站小程序
隨時(shí)隨地免費(fèi)學(xué)習(xí)課程

當(dāng)前位置:首頁  >  技術(shù)干貨  > Python淺談并發(fā)鎖與死鎖問題

Python淺談并發(fā)鎖與死鎖問題

來源:千鋒教育
發(fā)布人:xqq
時(shí)間: 2023-11-07 12:39:57 1699331997

如果你學(xué)過操作系統(tǒng),那么對(duì)于鎖應(yīng)該不陌生。鎖的含義是線程鎖,可以用來指定某一個(gè)邏輯或者是資源同一時(shí)刻只能有一個(gè)線程訪問。這個(gè)很好理解,就好像是有一個(gè)房間被一把鎖鎖住了,只有拿到鑰匙的人才能進(jìn)入。每一個(gè)人從房間門口拿到鑰匙進(jìn)入房間,出房間的時(shí)候會(huì)把鑰匙再放回到門口。這樣下一個(gè)到門口的人就可以拿到鑰匙了。這里的房間就是某一個(gè)資源或者是一段邏輯,而拿取鑰匙的人其實(shí)指的是一個(gè)線程。

加鎖的原因

我們明白了鎖的原理,不禁有了一個(gè)問題,我們?yōu)槭裁葱枰i呢,它在哪些場(chǎng)景當(dāng)中會(huì)用到呢?

其實(shí)它的使用場(chǎng)景非常廣,我們舉一個(gè)非常簡(jiǎn)單的例子,就是淘寶買東西。我們都知道商家的庫(kù)存都是有限的,賣掉一個(gè)少一個(gè)。假如說當(dāng)前某個(gè)商品庫(kù)存只剩下一個(gè),但當(dāng)下卻有兩個(gè)人同時(shí)購(gòu)買。兩個(gè)人同時(shí)購(gòu)買也就是有兩個(gè)請(qǐng)求同時(shí)發(fā)起購(gòu)買請(qǐng)求,如果我們不加鎖的話,兩個(gè)線程同時(shí)查詢到商品的庫(kù)存是1,大于0,進(jìn)行購(gòu)買邏輯之后,減一。由于兩個(gè)線程同時(shí)執(zhí)行,所以最后商品的庫(kù)存會(huì)變成-1。

顯然商品的庫(kù)存不應(yīng)該是一個(gè)負(fù)數(shù),所以我們需要避免這種情況發(fā)生。通過加鎖可以完美解決這個(gè)問題。我們規(guī)定一次只能有一個(gè)線程發(fā)起購(gòu)買的請(qǐng)求,那么這樣當(dāng)一個(gè)線程將庫(kù)存減到0的時(shí)候,第二個(gè)請(qǐng)求就無法修改了,就保證了數(shù)據(jù)的準(zhǔn)確性。

代碼實(shí)現(xiàn)

那么在Python當(dāng)中,我們?cè)趺礃觼韺?shí)現(xiàn)這個(gè)鎖呢?

其實(shí)很簡(jiǎn)單,threading庫(kù)當(dāng)中已經(jīng)為我們提供了線程的工具,我們直接拿過來用就可以了。我們通過使用threading當(dāng)中的Lock對(duì)象,可以很輕易的實(shí)現(xiàn)方法加鎖的功能。

importthreading

classPurchaseRequest:

'''

初始化庫(kù)存與鎖

'''

def__init__(self,initial_value=0):

self._value=initial_value

self._lock=threading.Lock()

defincr(self,delta=1):

'''

加庫(kù)存

'''

self._lock.acquire()

self._value+=delta

self._lock.release()

defdecr(self,delta=1):

'''

減庫(kù)存

'''

self._lock.acquire()

self._value-=delta

self._lock.release()

我們從代碼當(dāng)中就可以很輕易的看出Lock這個(gè)對(duì)象的使用方法,我們?cè)谶M(jìn)入加鎖區(qū)(資源搶占區(qū))之前,我們需要先使用lock.acquire()方法獲取鎖。Lock對(duì)象可以保證同一時(shí)刻只能有一個(gè)線程獲取鎖,只有獲取了鎖之后才會(huì)繼續(xù)往下執(zhí)行。當(dāng)我們執(zhí)行完成之后,我們需要把鎖“放回門口”,所以需要再調(diào)用一下release方法,表示鎖的釋放。

這里有一個(gè)小問題是很多程序員在編程的時(shí)候總是會(huì)忘記release,導(dǎo)致不必要的bug,而且這種分布式場(chǎng)景當(dāng)中的bug很難通過測(cè)試發(fā)現(xiàn)。因?yàn)闇y(cè)試的時(shí)候往往很難測(cè)試并發(fā)場(chǎng)景,codereview的時(shí)候也很容易忽略,因此一旦泄露了還是挺難發(fā)現(xiàn)的。

為了解決這個(gè)問題,Lock還提供了一種改進(jìn)的用法,就是使用with語句。with語句我們之前在使用文件的時(shí)候用到過,使用with可以替我們完成trycatch以及資源回收等工作,我們只管用就完事了。這里也是一樣,使用with之后我們就可以不用管鎖的申請(qǐng)和釋放了,直接寫代碼就行,所以上面的代碼可以改寫成這樣:

importthreading

classPurchaseRequest:

'''

初始化庫(kù)存與鎖

'''

def__init__(self,initial_value=0):

self._value=initial_value

self._lock=threading.Lock()

defincr(self,delta=1):

'''

加庫(kù)存

'''

withself._lock:

self._value+=delta

defdecr(self,delta=1):

'''

減庫(kù)存

'''

withself._lock:

self._value-=delta

這樣看起來是不是清爽很多?

可重入鎖

上面介紹的只是最簡(jiǎn)單的鎖,我們經(jīng)常使用的往往是可重入鎖。

什么叫可重入鎖呢?簡(jiǎn)單解釋一下,就是在一個(gè)線程已經(jīng)持有了鎖的情況下,它可以再次進(jìn)入被加鎖的區(qū)域。但是既然線程還持有鎖沒有釋放,那么它不應(yīng)該還是在加鎖區(qū)域嗎,怎么會(huì)有需要再次進(jìn)入被加鎖區(qū)域的情況呢?其實(shí)是有的,道理也很簡(jiǎn)單,就是遞歸。

我們把上面的例子稍微改一點(diǎn)點(diǎn),就完全不一樣了。

importthreading

classPurchaseRequest:

'''

初始化庫(kù)存與鎖

'''

def__init__(self,initial_value=0):

self._value=initial_value

self._lock=threading.Lock()

defincr(self,delta=1):

'''

加庫(kù)存

'''

withself._lock:

self._value+=delta

defdecr(self,delta=1):

'''

減庫(kù)存

'''

withself._lock:

self.incr(-delta)

我們關(guān)注一下上面的decr方法,我們用incr來代替了原本的邏輯實(shí)現(xiàn)了decr。但是有一個(gè)問題是decr也是一個(gè)加鎖的方法,需要前一個(gè)鎖釋放了才能進(jìn)入。但它已經(jīng)持有了鎖了,那么這種情況下就會(huì)發(fā)生死鎖。

我們只需要把Lock換成可重入鎖就可以解決這個(gè)問題,只需要修改一行代碼。

importthreading

classPurchaseRequest:

'''

初始化庫(kù)存與鎖

我們使用RLock代替了Lock,也可重入鎖代替了普通鎖

'''

def__init__(self,initial_value=0):

self._value=initial_value

self._lock=threading.RLock()

defincr(self,delta=1):

'''

加庫(kù)存

'''

withself._lock:

self._value+=delta

defdecr(self,delta=1):

'''

減庫(kù)存

'''

withself._lock:

self.incr(-delta)

以上內(nèi)容為大家介紹了Python淺談并發(fā)鎖與死鎖問題,希望對(duì)大家有所幫助,如果想要了解更多Python相關(guān)知識(shí),請(qǐng)關(guān)注IT培訓(xùn)機(jī)構(gòu):千鋒教育。

聲明:本站稿件版權(quán)均屬千鋒教育所有,未經(jīng)許可不得擅自轉(zhuǎn)載。
10年以上業(yè)內(nèi)強(qiáng)師集結(jié),手把手帶你蛻變精英
請(qǐng)您保持通訊暢通,專屬學(xué)習(xí)老師24小時(shí)內(nèi)將與您1V1溝通
免費(fèi)領(lǐng)取
今日已有369人領(lǐng)取成功
劉同學(xué) 138****2860 剛剛成功領(lǐng)取
王同學(xué) 131****2015 剛剛成功領(lǐng)取
張同學(xué) 133****4652 剛剛成功領(lǐng)取
李同學(xué) 135****8607 剛剛成功領(lǐng)取
楊同學(xué) 132****5667 剛剛成功領(lǐng)取
岳同學(xué) 134****6652 剛剛成功領(lǐng)取
梁同學(xué) 157****2950 剛剛成功領(lǐng)取
劉同學(xué) 189****1015 剛剛成功領(lǐng)取
張同學(xué) 155****4678 剛剛成功領(lǐng)取
鄒同學(xué) 139****2907 剛剛成功領(lǐng)取
董同學(xué) 138****2867 剛剛成功領(lǐng)取
周同學(xué) 136****3602 剛剛成功領(lǐng)取
相關(guān)推薦HOT
主站蜘蛛池模板: 通辽市| 丰台区| 昆明市| 亳州市| 长岛县| 赤壁市| 微山县| 舞钢市| 邢台市| 修文县| 重庆市| 伊川县| 大洼县| 泸溪县| 浪卡子县| 萝北县| 霍城县| 乌拉特中旗| 大洼县| 泊头市| 财经| 策勒县| 陇南市| 措美县| 桃源县| 宁德市| 行唐县| 攀枝花市| 城固县| 全南县| 庆安县| 台山市| 乌拉特后旗| 田东县| 汾西县| 丰镇市| 唐河县| 浠水县| 铜陵市| 威海市| 交城县|