99999久久久久久亚洲,欧美人与禽猛交狂配,高清日韩av在线影院,一个人在线高清免费观看,啦啦啦在线视频免费观看www

熱線電話:13121318867

登錄
首頁精彩閱讀舉例講解Python中的死鎖、可重入鎖和互斥鎖
舉例講解Python中的死鎖、可重入鎖和互斥鎖
2018-03-18
收藏

舉例講解Python中的死鎖、可重入鎖和互斥鎖

一、死鎖

簡單來說,死鎖是一個(gè)資源被多次調(diào)用,而多次調(diào)用方都未能釋放該資源就會(huì)造成死鎖,這里結(jié)合例子說明下兩種常見的死鎖情況。

1、迭代死鎖

該情況是一個(gè)線程“迭代”請求同一個(gè)資源,直接就會(huì)造成死鎖:

import threading
import time
class MyThread(threading.Thread):
  def run(self):
    global num
    time.sleep(1)
    if mutex.acquire(1):
      num = num+1
      msg = self.name+' set num to '+str(num)
      print msg
      mutex.acquire()
      mutex.release()
      mutex.release()
num = 0
mutex = threading.Lock()
def test():
  for i in range(5):
    t = MyThread()
    t.start()
if __name__ == '__main__':
  test()


上例中,在run函數(shù)的if判斷中第一次請求資源,請求后還未 release ,再次acquire,最終無法釋放,造成死鎖。這里例子中通過將print下面的兩行注釋掉就可以正常執(zhí)行了 ,除此之外也可以通過可重入鎖解決,后面會(huì)提到。

2、互相調(diào)用死鎖

上例中的死鎖是在同一個(gè)def函數(shù)內(nèi)多次調(diào)用造成的,另一種情況是兩個(gè)函數(shù)中都會(huì)調(diào)用相同的資源,互相等待對方結(jié)束的情況。如果兩個(gè)線程分別占有一部分資源并且同時(shí)等待對方的資源,就會(huì)造成死鎖。

import threading
import time
class MyThread(threading.Thread):
  def do1(self):
    global resA, resB
    if mutexA.acquire():
       msg = self.name+' got resA'
       print msg
       if mutexB.acquire(1):
         msg = self.name+' got resB'
         print msg
         mutexB.release()
       mutexA.release()
  def do2(self):
    global resA, resB
    if mutexB.acquire():
       msg = self.name+' got resB'
       print msg
       if mutexA.acquire(1):
         msg = self.name+' got resA'
         print msg
         mutexA.release()
       mutexB.release()
  def run(self):
    self.do1()
    self.do2()
resA = 0
resB = 0
mutexA = threading.Lock()
mutexB = threading.Lock()
def test():
  for i in range(5):
    t = MyThread()
    t.start()
if __name__ == '__main__':
  test()



這個(gè)死鎖的示例稍微有點(diǎn)復(fù)雜。具體可以理下。

二、可重入鎖

為了支持在同一線程中多次請求同一資源,python提供了“可重入鎖”:threading.RLock。RLock內(nèi)部維護(hù)著一個(gè)Lock和一個(gè)counter變量,counter記錄了acquire的次數(shù),從而使得資源可以被多次require。直到一個(gè)線程所有的acquire都被release,其他的線程才能獲得資源。這里以例1為例,如果使用RLock代替Lock,則不會(huì)發(fā)生死鎖:

import threading
import time
class MyThread(threading.Thread):
  def run(self):
    global num
    time.sleep(1)
    if mutex.acquire(1):
      num = num+1
      msg = self.name+' set num to '+str(num)
      print msg
      mutex.acquire()
      mutex.release()
      mutex.release()
num = 0
mutex = threading.RLock()
def test():
  for i in range(5):
    t = MyThread()
    t.start()
if __name__ == '__main__':
  test()



和上面那個(gè)例子的不同之處在于threading.Lock()換成了threading.RLock() 。

三、互斥鎖
python threading模塊有兩類鎖:互斥鎖(threading.Lock )和可重用鎖(threading.RLock)。兩者的用法基本相同,具體如下:

lock = threading.Lock()
lock.acquire()
dosomething……
lock.release()


RLock的用法是將threading.Lock()修改為threading.RLock()。便于理解,先來段代碼:

[root@361way lock]# cat lock1.py
    
#!/usr/bin/env python
# coding=utf-8
import threading              # 導(dǎo)入threading模塊
import time               # 導(dǎo)入time模塊
class mythread(threading.Thread):    # 通過繼承創(chuàng)建類
  def __init__(self,threadname):   # 初始化方法
    # 調(diào)用父類的初始化方法
    threading.Thread.__init__(self,name = threadname)
  def run(self):             # 重載run方法
    global x         # 使用global表明x為全局變量
    for i in range(3):
      x = x + 1
    time.sleep(5)     # 調(diào)用sleep函數(shù),讓線程休眠5秒
    print x
tl = []               # 定義列表
for i in range(10):
  t = mythread(str(i))        # 類實(shí)例化
  tl.append(t)           # 將類對象添加到列表中
x=0                 # 將x賦值為0
for i in tl:
  i.start()
這里執(zhí)行的結(jié)果和想想的不同,結(jié)果如下:    
[root@361way lock]# python lock1.py
    
30
30
30
30
30
30
30
30
30
30


為什么結(jié)果都是30呢?關(guān)鍵在于global 行和 time.sleep行。

1、由于x是一個(gè)全局變量,所以每次循環(huán)后 x 的值都是執(zhí)行后的結(jié)果值;

2、由于該代碼是多線程的操作,所以在sleep 等待的時(shí)候,之前已經(jīng)執(zhí)行完成的線程會(huì)在這等待,而后續(xù)的進(jìn)程在等待的5秒這段時(shí)間也執(zhí)行完成 ,等待print。同樣由于global 的原理,x被重新斌值。所以打印出的結(jié)果全是30 ;

3、便于理解,可以嘗試將sleep等注釋,你再看下結(jié)果,就會(huì)發(fā)現(xiàn)有不同。

在實(shí)際應(yīng)用中,如抓取程序等,也會(huì)出現(xiàn)類似于sleep等待的情況。在前后調(diào)用有順序或打印有輸出的時(shí)候,就會(huì)現(xiàn)并發(fā)競爭,造成結(jié)果或輸出紊亂。這里就引入了鎖的概念,上面的代碼修改下,如下:

[root@361way lock]# cat lock2.py
    
#!/usr/bin/env python
# coding=utf-8
import threading              # 導(dǎo)入threading模塊
import time               # 導(dǎo)入time模塊
class mythread(threading.Thread):          # 通過繼承創(chuàng)建類
  def __init__(self,threadname):         # 初始化方法
    threading.Thread.__init__(self,name = threadname)
  def run(self):             # 重載run方法
    global x            # 使用global表明x為全局變量
    lock.acquire()           # 調(diào)用lock的acquire方法
    for i in range(3):
      x = x + 1
    time.sleep(5)      # 調(diào)用sleep函數(shù),讓線程休眠5秒
    print x
    lock.release()        # 調(diào)用lock的release方法
lock = threading.Lock()        # 類實(shí)例化
tl = []             # 定義列表
for i in range(10):
  t = mythread(str(i))      # 類實(shí)例化
  tl.append(t)       # 將類對象添加到列表中
x=0            # 將x賦值為0
for i in tl:
  i.start()           # 依次運(yùn)行線程

執(zhí)行的結(jié)果如下:
    
[root@361way lock]# python lock2.py
    
3
6
9
12
15
18
21
24
27
30


加鎖的結(jié)果會(huì)造成阻塞,而且會(huì)造成開鎖大。會(huì)根據(jù)順序由并發(fā)的多線程按順序輸出,如果后面的線程執(zhí)行過快,需要等待前面的進(jìn)程結(jié)束后其才能結(jié)束 --- 寫的貌似有點(diǎn)像隊(duì)列的概念了 ,不過在加鎖的很多場景下確實(shí)可以通過隊(duì)列去解決。


數(shù)據(jù)分析咨詢請掃描二維碼

若不方便掃碼,搜微信號(hào):CDAshujufenxi

數(shù)據(jù)分析師資訊
更多

OK
客服在線
立即咨詢
客服在線
立即咨詢
') } function initGt() { var handler = function (captchaObj) { captchaObj.appendTo('#captcha'); captchaObj.onReady(function () { $("#wait").hide(); }).onSuccess(function(){ $('.getcheckcode').removeClass('dis'); $('.getcheckcode').trigger('click'); }); window.captchaObj = captchaObj; }; $('#captcha').show(); $.ajax({ url: "/login/gtstart?t=" + (new Date()).getTime(), // 加隨機(jī)數(shù)防止緩存 type: "get", dataType: "json", success: function (data) { $('#text').hide(); $('#wait').show(); // 調(diào)用 initGeetest 進(jìn)行初始化 // 參數(shù)1:配置參數(shù) // 參數(shù)2:回調(diào),回調(diào)的第一個(gè)參數(shù)驗(yàn)證碼對象,之后可以使用它調(diào)用相應(yīng)的接口 initGeetest({ // 以下 4 個(gè)配置參數(shù)為必須,不能缺少 gt: data.gt, challenge: data.challenge, offline: !data.success, // 表示用戶后臺(tái)檢測極驗(yàn)服務(wù)器是否宕機(jī) new_captcha: data.new_captcha, // 用于宕機(jī)時(shí)表示是新驗(yàn)證碼的宕機(jī) product: "float", // 產(chǎn)品形式,包括:float,popup width: "280px", https: true // 更多配置參數(shù)說明請參見:http://docs.geetest.com/install/client/web-front/ }, handler); } }); } function codeCutdown() { if(_wait == 0){ //倒計(jì)時(shí)完成 $(".getcheckcode").removeClass('dis').html("重新獲取"); }else{ $(".getcheckcode").addClass('dis').html("重新獲取("+_wait+"s)"); _wait--; setTimeout(function () { codeCutdown(); },1000); } } function inputValidate(ele,telInput) { var oInput = ele; var inputVal = oInput.val(); var oType = ele.attr('data-type'); var oEtag = $('#etag').val(); var oErr = oInput.closest('.form_box').next('.err_txt'); var empTxt = '請輸入'+oInput.attr('placeholder')+'!'; var errTxt = '請輸入正確的'+oInput.attr('placeholder')+'!'; var pattern; if(inputVal==""){ if(!telInput){ errFun(oErr,empTxt); } return false; }else { switch (oType){ case 'login_mobile': pattern = /^1[3456789]\d{9}$/; if(inputVal.length==11) { $.ajax({ url: '/login/checkmobile', type: "post", dataType: "json", data: { mobile: inputVal, etag: oEtag, page_ur: window.location.href, page_referer: document.referrer }, success: function (data) { } }); } break; case 'login_yzm': pattern = /^\d{6}$/; break; } if(oType=='login_mobile'){ } if(!!validateFun(pattern,inputVal)){ errFun(oErr,'') if(telInput){ $('.getcheckcode').removeClass('dis'); } }else { if(!telInput) { errFun(oErr, errTxt); }else { $('.getcheckcode').addClass('dis'); } return false; } } return true; } function errFun(obj,msg) { obj.html(msg); if(msg==''){ $('.login_submit').removeClass('dis'); }else { $('.login_submit').addClass('dis'); } } function validateFun(pat,val) { return pat.test(val); }