多線程 -- threading

多線程模塊 threading

  • 創(chuàng)建多線程的兩種方式:

    
    import threading
    import time
    
    
    
  • 創(chuàng)建線程的第一種方式

    def foo(n):
        pass
    print('hello world')
        
    t1 = threading.Thread(target=foo, args=(1,))
    t1.start()
    
  • 創(chuàng)建線程的第二種方式

``` 
class MyThread(threading.Thread):

    def __init__(self, num):
        threading.Thread.__init__(self)
        self.num = num   # 對象里的字段  # self 是 t1或t2

    def run(self):

        print("running on number: %s" % self.num)

        time.sleep(3)

if __name__ == '__main__':
    t1 = MyThread(1)  # 實(shí)例化一個類的對象,執(zhí)行構(gòu)造方法
    t2 = MyThread(2)

    t1.start()
    t2.start()
``` 
  • 創(chuàng)建線程:
    import threading
    t1 = threading.Thread(target=music, args=('七里香',))
    
    target指向函數(shù),args指向函數(shù)的參數(shù)
  • 啟動線程:

      t.start()
    
  • 阻塞線程:

      t.join()    阻塞直到線程t結(jié)束   
    
  • 多線程實(shí)例1

      import threading
      from time import ctime, sleep
      import time
          
      # 預(yù)計耗時4秒
      def music(func):
          for i in range(2):
              print("I was listening to %s. %s" % (func, ctime()))
              sleep(2)
              print("end listening %s" %ctime())
      
      # 預(yù)計耗時6秒
      def movie(func):
          for i in range(2):
              print("Begin watching at the %s! %s" % (func,ctime()))
              sleep(3)
              print("end watching %s" % ctime())
      
      
      # 創(chuàng)建線程1,2
      
      threads = []
      t1 = threading.Thread(target=music, args=('七里香',))
      threads.append(t1)
      t2 = threading.Thread(target=movie, args=('十三區(qū)',))
      threads.append(t2)
      
      
      if __name__ == '__main__':
          for t in threads:
              t.start()
          start = time.time()
          t2.join()
          end = time.time()
      
      
          # t2.join()    # join阻塞
          print("總共耗時 %s" % (end-start))
          
          
      # 結(jié)果:總共耗時 6.004828929901123
    
  • 多線程 -- 計算密集型

    import threading
    import time
    
    begin = time.time()
    
    def add(n):
        sum = 0
        for i in range(n):
            sum += i
        print(sum)
    
    
    # 單線程串行計算
    # add(10000000)
    # add(10000000)
    
    
    # 多線程并行計算
    t1 = threading.Thread(target=add, args=(10000000,))
    t1.start()
    # print('1開始了')
    
    t2 = threading.Thread(target=add, args=(10000000,))
    t2.start()
    # print('2也開始了')
    
    t1.join()
    t2.join()
    #
    end = time.time()
    
    print(end-begin)
    
    
    # 串行花費(fèi) 14.73 秒  兩者都是計算密集型,都在搶CPU
    # 并行花費(fèi) 15.71 秒  并行反而更多,因?yàn)榍袚Q還需要時間
    
    # IO密集型任務(wù)或函數(shù)
    # 計算密集型任務(wù)
    
    # GIL全局解釋鎖  如果只有一顆CPU,無法解決計算密集型的任務(wù) 只有Cpython有這個bug
    
  • 多線程 -- IO密集型

    import time
    import threading
    
    
    def foo(n):                         # 設(shè)定3秒
        print('線程1開始')
        print('foo %s' % n)
        # print('foo')
        time.sleep(3)
        print('線程1結(jié)束\n')
    
    
    def bar(n):                         # 設(shè)定1秒
        print('線程2開始')
        print('bar %s' % n)
        time.sleep(1)
        # print('ok')
        print('線程2結(jié)束')
    
    
    # t1 t2 即為線程
    # target = 要執(zhí)行的函數(shù) args = 參數(shù)
    t1 = threading.Thread(target=foo, args=(1,))
    t2 = threading.Thread(target=bar, args=(2,))
    
    print('...... in the main  ........')
    
    # 線程執(zhí)行 start()
    
    begin = time.time()
    
    t1.start()  # 線程開始
    t2.start()  # 線程開始
    
    t1.join()  # 子線程不結(jié)束不往下走
    t2.join()  # 子線程不結(jié)束不往下走
    
    end = time.time()
    print('整個程序一共花了', end - begin, '秒,asshole')  #
    
    
    # 運(yùn)行結(jié)果: 整個程序一共花了 3.003476858139038 秒,asshole
        
    
  • 死鎖

    # 死鎖怎么辦?
    # 用遞歸鎖解決死鎖問題,遞歸鎖可重用
    
    import threading, time
    
    class myThread(threading.Thread):
    
    def doA(self):
        # lockA.acquire()
        lock.acquire()                  # 加鎖        # 為什么在外面加了一把鎖后還要在里面再加一把鎖
        print(self.name, "gotlockA", time.ctime())
        time.sleep(1)
        # lockB.acquire()
        lock.acquire()                  # 加第二把鎖
        print(self.name, "gotlockB", time.ctime())
        # lockB.release()
        lock.release()                  # 釋放第一把鎖
        # lockA.release()
        lock.release()                  # 釋放第二把鎖
    
    def doB(self):
        lock.acquire()
        print(self.name, "gotlockB", time.ctime())
        time.sleep(1)
        lock.acquire()
        print(self.name, "gotlockA", time.ctime())
        lock.release()
        lock.release()
    
    def run(self):   # 類似于構(gòu)造方法
        self.doA()
        self.doB()
    
    
    if __name__ == "__main__":
    
        # lockA = threading.Lock()     # 普通鎖
        # lockB = threading.Lock()
        lock = threading.RLock()       # Rlock 叫遞歸鎖,可重用
        threads = []
        for i in range(5):
            threads.append(myThread())
        for t in threads:
            t.start()
        for t in threads:
            t.join()  # 等待線程結(jié)束,后面再講。
    
    
  • 同步鎖

        # 加鎖后會加鎖的部分變成串行
    
    import time
    import threading
    
    def addNum():
        global num  # 在每個線程中都獲取這個變量
        print('fuck off')
        # num -= 1          #  此操作的時間遠(yuǎn)小于CPU切換的時間
    
        # 加鎖,被加鎖的部分是串行
    
        r.acquire()         #######  加鎖
        temp = num          # 1
        # time.sleep(0.001)   # 2 有可能取了數(shù)99后,執(zhí)行完這兩步后CPU轉(zhuǎn)到另外的線程上去了,
        print('ok')                    # 另外的線程取到99,執(zhí)行減1,CPU再轉(zhuǎn)回當(dāng)前線程,又執(zhí)行99-1,重復(fù)了
        time.sleep(0.0002)
        num = temp - 1
        r.release()         ######## 釋放鎖
    
    
    
    num = 100
    thread_list = []
    
    r = threading.Lock()    # python的GIL鎖
    
    # 用for循環(huán)創(chuàng)建100個線程
    for i in range(100):
        t = threading.Thread(target=addNum)
        t.start()
        # t.join()     # join阻塞,可以得到正確的結(jié)果,從但是變成了串行,失去了多線程的意義
                       # join雖然可以得到正確結(jié)果,但是除了計算的其他部分也會變成并行
        thread_list.append(t)
    
    
    for t in thread_list:
        t.join()
    
    
    print("\nfinal num:", num)
    
    
    # num減1的速度如果
    
    # GIL保證只有一個線程進(jìn)入解釋器
    # 自己加的鎖是鎖CPU,在我釋放鎖之前你不要切換了
    
    
  • event

    # event 類似condition
    
    # event.isSet():返回event的狀態(tài)值;
    #
    # event.wait():如果 event.isSet()==False將阻塞線程;
    #
    # event.set(): 設(shè)置event的狀態(tài)值為True,所有阻塞池的線程激活進(jìn)入就緒狀態(tài), 等待操作系統(tǒng)調(diào)度;
    #
    # event.clear():恢復(fù)event的狀態(tài)值為False。
    
    
    import threading, time
    
    class Boss(threading.Thread):
        def run(self):
            print("BOSS: 今晚大家都要加班到22:00")
            event.is_set() or event.set()
            time.sleep(2)
            print("BOSS:<22:00>可以下班了。 ")
            event.is_set() or event.set()
    
    class Worker(threading.Thread):
        def run(self):
            event.wait()
            print("Worker:哎,命苦啊")
            time.sleep(0.25)
            event.clear()
            event.wait()
            print("Worker: OhYear!")
    
    
    
    
    
    if __name__ == "__main__":
        event = threading.Event()
        threads = []
        for i in range(5):
            threads.append(Worker())
        threads.append(Boss())
        for t in threads:
            t.start()
        for t in threads:
            t.join()
    
    
  • 多線程利器 QUEUE隊列

        # 隊列里本身就有一把鎖
    #
    # import queue, time, threading
    #
    # d = queue.Queue(3)   #
    
    #  d.put('jinxing',0)  #插入數(shù)據(jù)
    # d.put('xiaohu')
    # d.put('haoran')
    #
    # print(d.get())
    # print(d.get())
    # print(d.get())
    #
    #
    # print(d.qsize())
    
    # li = [1,2,3,4,5]
    #
    # def pri():
    #     while li:
    #         a = li[-1]
    #         print(a)
    #         time.sleep(1)
    #         try:
    #             li.remove(a)
    #         except:
    #             print('----',a)
    #
    # t1 = threading.Thread(target=pri, args=())
    # t1.start()
    # t2 = threading.Thread(target=pri, args=())
    # t2.start()
    
    import threading, queue
    from time import sleep
    from random import randint
    
    
    class Production(threading.Thread):   #  生產(chǎn)者
        def run(self):
            while True:
                r = randint(0, 100)
                q.put(r)
                print("生產(chǎn)出來%s號包子" % r)
                sleep(1)
    
    
    class Proces(threading.Thread):       #  消費(fèi)者
        def run(self):
            while True:
                re = q.get()
                print("吃掉%s號包子" % re)
                sleep(1)
    
    
    if __name__ == "__main__":
        q = queue.Queue(10)
        threads = [Production(), Production(), Production(), Proces()]
        for t in threads:
            t.start()
    
    
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,702評論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,143評論 3 415
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,553評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,620評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,416評論 6 405
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 54,940評論 1 321
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,024評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,170評論 0 287
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,709評論 1 333
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,597評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,784評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,291評論 5 357
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,029評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,407評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,663評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,403評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,746評論 2 370

推薦閱讀更多精彩內(nèi)容

  • 線程狀態(tài)新建,就緒,運(yùn)行,阻塞,死亡。 線程同步多線程可以同時運(yùn)行多個任務(wù),線程需要共享數(shù)據(jù)的時候,可能出現(xiàn)數(shù)據(jù)不...
    KevinCool閱讀 814評論 0 0
  • 我們前面提到了進(jìn)程是由若干線程組成的,一個進(jìn)程至少有一個線程。多線程優(yōu)點(diǎn): 在一個進(jìn)程中的多線程和主線程分享相同的...
    第八共同體閱讀 525評論 0 0
  • 線程 引言&動機(jī) 考慮一下這個場景,我們有10000條數(shù)據(jù)需要處理,處理每條數(shù)據(jù)需要花費(fèi)1秒,但讀取數(shù)據(jù)只需要0....
    不浪漫的浪漫_ea03閱讀 367評論 0 0
  • 引言&動機(jī) 考慮一下這個場景,我們有10000條數(shù)據(jù)需要處理,處理每條數(shù)據(jù)需要花費(fèi)1秒,但讀取數(shù)據(jù)只需要0.1秒,...
    chen_000閱讀 516評論 0 0
  • 1.進(jìn)程和線程 隊列:1、進(jìn)程之間的通信: q = multiprocessing.Queue()2、...
    一只寫程序的猿閱讀 1,118評論 0 17