python 線程的暫停, 恢複, 退出詳解及執行個體_python

來源:互聯網
上載者:User

python 線程 暫停, 恢複, 退出

我們都知道python中可以是threading模組實現多線程, 但是模組並沒有提供暫停, 恢複和停止線程的方法, 一旦線程對象調用start方法後, 只能等到對應的方法函數運行完畢. 也就是說一旦start後, 線程就屬於失控狀態. 不過, 我們可以自己實現這些. 一般的方法就是迴圈地判斷一個標誌位, 一旦標誌位到達到預定的值, 就退出迴圈. 這樣就能做到退出線程了. 但暫停和恢複線程就有點難了, 我一直也不清除有什麼好的方法, 直到我看到threading中Event對象的wait方法的描述時.

wait([timeout])  Block until the internal flag is true. If the internal flag is true on entry, return immediately. Otherwise, block until another thread calls set() to set the flag to true, or until the optional timeout occurs.  阻塞, 直到內部的標誌位為True時. 如果在內部的標誌位在進入時為True時, 立即返回. 否則, 阻塞直到其他線程調用set()方法將標準位設為True, 或者到達了可選的timeout時間.  When the timeout argument is present and not None, it should be a floating point number specifying a timeout for the operation in seconds (or fractions thereof).  This method returns the internal flag on exit, so it will always return True except if a timeout is given and the operation times out.  當給定了timeout參數且不為None, 它應該是一個浮點數,以秒為單位指定操作的逾時(或是分數)。  此方法在退出時返回內部標誌,因此除非給定了逾時且操作逾時,否則它將始終返回True。  Changed in version 2.7: Previously, the method always returned None.  2.7版本以前, 這個方法總會返回None.

  利用wait的阻塞機制, 就能夠實現暫停和恢複了, 再配合迴圈判斷標識位, 就能實現退出了, 下面是程式碼範例:

#!/usr/bin/env python# coding: utf-8import threadingimport timeclass Job(threading.Thread):  def __init__(self, *args, **kwargs):    super(Job, self).__init__(*args, **kwargs)    self.__flag = threading.Event()   # 用於暫停線程的標識    self.__flag.set()    # 設定為True    self.__running = threading.Event()   # 用於停止線程的標識    self.__running.set()   # 將running設定為True  def run(self):    while self.__running.isSet():      self.__flag.wait()   # 為True時立即返回, 為False時阻塞直到內部的標識位為True後返回      print time.time()      time.sleep(1)  def pause(self):    self.__flag.clear()   # 設定為False, 讓線程阻塞  def resume(self):    self.__flag.set()  # 設定為True, 讓線程停止阻塞  def stop(self):    self.__flag.set()    # 將線程從暫停狀態恢複, 如何已經暫停話    self.__running.clear()    # 設定為False  

下面是測試代碼:

a = Job()a.start()time.sleep(3)a.pause()time.sleep(3)a.resume()time.sleep(3)a.pause()time.sleep(2)a.stop()

測試的結果:

 

  這完成了暫停, 恢複和停止的功能. 但是這裡有一個缺點: 無論是暫停還是停止, 都不是瞬時的, 必須等待run函數內部的運行到達標誌位判斷時才有效. 也就是說操作會滯後一次.

  但是這有時也不一定是壞事. 如果run函數中涉及了檔案操作或資料庫操作等, 完整地運行一次後再退出, 反而能夠執行剩餘的資源釋放操作的代碼(例如各種close). 不會出現程式的檔案操作符超出上限, 資料庫連接未釋放等尷尬的情況.

感謝閱讀,希望能協助到大家,謝謝大家對本站的支援!

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.