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

Python中多線程及程序鎖淺析

 更新時(shí)間:2015年01月21日 11:25:18   投稿:junjie  
這篇文章主要介紹了Python中多線程及程序鎖淺析,本文用一個(gè)實(shí)例講解Python的多線程和程序鎖,需要的朋友可以參考下

Python中多線程使用到Threading模塊。Threading模塊中用到的主要的類是Thread,我們先來寫一個(gè)簡(jiǎn)單的多線程代碼:

復(fù)制代碼 代碼如下:

# coding : uft-8
__author__ = 'Phtih0n'
import threading

class MyThread(threading.Thread):
    def __init__(self):
        threading.Thread.__init__(self)

    def run(self):
        global n
        print n
        n += 1

if "__main__" == __name__:
    n = 0
    ThreadList = []
    for i in range(0, 10):
        t = MyThread()
        ThreadList.append(t)
    for t in ThreadList:
        t.start()
    for t in ThreadList:
        t.join

最普通的一個(gè)多線程小例子。我一筆帶過地講一講,我創(chuàng)建了一個(gè)繼承Thread類的子類MyThread,作為我們的線程啟動(dòng)類。按照規(guī)定,重寫Thread的run方法,我們的線程啟動(dòng)起來后會(huì)自動(dòng)調(diào)用該方法。于是我首先創(chuàng)建了10個(gè)線程,并將其加入列表中。再使用一個(gè)for循環(huán),開啟每個(gè)線程。在使用一個(gè)for循環(huán),調(diào)用join方法等待所有線程結(jié)束才退出主線程。

這段代碼看似簡(jiǎn)單,但實(shí)際上隱藏著一個(gè)很大的問題,只是在這里沒有體現(xiàn)出來。你真的以為我創(chuàng)建了10個(gè)線程,并按順序調(diào)用了這10個(gè)線程,每個(gè)線程為n增加了1.實(shí)際上,有可能是A線程執(zhí)行了n++,再C線程執(zhí)行了n++,再B線程執(zhí)行n++。

這里涉及到一個(gè)“鎖”的問題,如果有多個(gè)線程同時(shí)操作一個(gè)對(duì)象,如果沒有很好地保護(hù)該對(duì)象,會(huì)造成程序結(jié)果的不可預(yù)期(比如我們?cè)诿總€(gè)線程的run方法中加入一個(gè)time.sleep(1),并同時(shí)輸出線程名稱,則我們會(huì)發(fā)現(xiàn),輸出會(huì)亂七八糟。因?yàn)榭赡芪覀兊囊粋€(gè)print語(yǔ)句只打印出一半的字符,這個(gè)線程就被暫停,執(zhí)行另一個(gè)去了,所以我們看到的結(jié)果很亂),這種現(xiàn)象叫做“線程不安全”:

于是,Threading模塊為我們提供了一個(gè)類,Threading.Lock,鎖。我們創(chuàng)建一個(gè)該類對(duì)象,在線程函數(shù)執(zhí)行前,“搶占”該鎖,執(zhí)行完成后,“釋放”該鎖,則我們確保了每次只有一個(gè)線程占有該鎖。這時(shí)候?qū)σ粋€(gè)公共的對(duì)象進(jìn)行操作,則不會(huì)發(fā)生線程不安全的現(xiàn)象了。

于是,我們把代碼更改如下:

復(fù)制代碼 代碼如下:

# coding : uft-8
__author__ = 'Phtih0n'
import threading, time

class MyThread(threading.Thread):
    def __init__(self):
        threading.Thread.__init__(self)

    def run(self):
        global n, lock
        time.sleep(1)
        if lock.acquire():
            print n , self.name
            n += 1
            lock.release()

if "__main__" == __name__:
    n = 1
    ThreadList = []
    lock = threading.Lock()
    for i in range(1, 200):
        t = MyThread()
        ThreadList.append(t)
    for t in ThreadList:
        t.start()
    for t in ThreadList:
        t.join()

最后執(zhí)行結(jié)果:

我們看到,我們先建立了一個(gè)threading.Lock類對(duì)象lock,在run方法里,我們使用lock.acquire()獲得了這個(gè)鎖。此時(shí),其他的線程就無法再獲得該鎖了,他們就會(huì)阻塞在“if lock.acquire()”這里,直到鎖被另一個(gè)線程釋放:lock.release()。

所以,if語(yǔ)句中的內(nèi)容就是一塊完整的代碼,不會(huì)再存在執(zhí)行了一半就暫停去執(zhí)行別的線程的情況。所以最后結(jié)果是整齊的。

就如同在java中,我們使用synchronized關(guān)鍵字修飾一個(gè)方法,目的一樣,讓某段代碼被一個(gè)線程執(zhí)行時(shí),不會(huì)打斷跳到另一個(gè)線程中。

這是多線程占用一個(gè)公共對(duì)象時(shí)候的情況。如果多個(gè)線程要調(diào)用多個(gè)現(xiàn)象,而A線程調(diào)用A鎖占用了A對(duì)象,B線程調(diào)用了B鎖占用了B對(duì)象,A線程不能調(diào)用B對(duì)象,B線程不能調(diào)用A對(duì)象,于是一直等待。這就造成了線程“死鎖”。

Threading模塊中,也有一個(gè)類,RLock,稱之為可重入鎖。該鎖對(duì)象內(nèi)部維護(hù)著一個(gè)Lock和一個(gè)counter對(duì)象。counter對(duì)象記錄了acquire的次數(shù),使得資源可以被多次require。最后,當(dāng)所有RLock被release后,其他線程才能獲取資源。在同一個(gè)線程中,RLock.acquire可以被多次調(diào)用,利用該特性,可以解決部分死鎖問題。

死鎖問題很復(fù)雜,多年來人們想出了很多算法來解決它。我就不再多說,具體還是要大家參閱幫助文檔。

相關(guān)文章

  • Python中Playwright的常用操作方法分享

    Python中Playwright的常用操作方法分享

    本文詳細(xì)介紹了Playwright的常用操作方法,包括獲取頁(yè)面元素、點(diǎn)擊按鈕和鏈接等。這些方法可以幫助開發(fā)者更加高效地進(jìn)行自動(dòng)化測(cè)試和爬蟲開發(fā),需要的可以參考一下
    2023-05-05
  • python將原圖裁剪為固定尺寸小圖

    python將原圖裁剪為固定尺寸小圖

    這篇文章主要為大家詳細(xì)介紹了python將原圖裁剪為固定尺寸小圖,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-04-04
  • Python scipy的二維圖像卷積運(yùn)算與圖像模糊處理操作示例

    Python scipy的二維圖像卷積運(yùn)算與圖像模糊處理操作示例

    這篇文章主要介紹了Python scipy的二維圖像卷積運(yùn)算與圖像模糊處理操作,涉及Python數(shù)學(xué)運(yùn)算與圖形繪制相關(guān)操作技巧,需要的朋友可以參考下
    2019-09-09
  • Django web框架使用url path name詳解

    Django web框架使用url path name詳解

    這篇文章主要介紹了Django web框架使用url path name詳解,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2019-04-04
  • 詳解Django rest_framework實(shí)現(xiàn)RESTful API

    詳解Django rest_framework實(shí)現(xiàn)RESTful API

    這篇文章主要介紹了詳解Django rest_framework實(shí)現(xiàn)RESTful API,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-05-05
  • python 三邊測(cè)量定位的實(shí)現(xiàn)代碼

    python 三邊測(cè)量定位的實(shí)現(xiàn)代碼

    這篇文章主要介紹了python 三邊測(cè)量定位的實(shí)現(xiàn)代碼,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • Python進(jìn)度條tqdm的用法詳解

    Python進(jìn)度條tqdm的用法詳解

    這篇文章主要介紹了Python進(jìn)度條tqdm的用法,這對(duì)于第三方庫(kù)非常豐富的Python來說,想要實(shí)現(xiàn)這一功能并不是什么難事,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),感興趣的朋友一起看看吧
    2021-08-08
  • Python用模塊pytz來轉(zhuǎn)換時(shí)區(qū)

    Python用模塊pytz來轉(zhuǎn)換時(shí)區(qū)

    在Python中,與時(shí)間相關(guān)的庫(kù)有好些,可以幫助我們快速的處理與時(shí)間相關(guān)的需求和問題。這里想和大家分享一下如何在Python用模塊pytz來轉(zhuǎn)換時(shí)區(qū)。
    2016-08-08
  • tensorflow 獲取checkpoint中的變量列表實(shí)例

    tensorflow 獲取checkpoint中的變量列表實(shí)例

    今天小編就為大家分享一篇tensorflow 獲取checkpoint中的變量列表實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2020-02-02
  • python引入其他py文件或模塊

    python引入其他py文件或模塊

    本文主要介紹了python引入其他py文件或模塊,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-01-01

最新評(píng)論