1、引入event。python
每一個線程,都是一個獨立運行的個體,而且每一個線程的運行狀態是沒法預測的。
redis
若是一個程序中有不少個線程,程序的其餘線程須要判斷某個線程的運行狀態,來肯定本身下一步要執行哪些操做。服務器
threading模塊中的event對象剛好能作到這一點,event對象包含了一個能夠經過線程設置的一個信號標誌位,它容許線程一直等待某些事件的發生。python2.7
在初始化默認的狀況下,event對象中的信號標識被設置爲「假」,若是這時,有一個線程等待這個event對象,而這個event對象的信號標誌爲「假」,那麼這個線程就會被一直阻塞下去,直到這個event信號標誌爲「真」,全部等待這個event對象的線程纔會被喚醒。
ide
也就是說,一個線程若是等待的event對象中的信號標誌位「真」,這個線程會忽略這個事件(event),繼續執行。函數
2、event對象的經常使用方法。spa
event.isSet():返回event的狀態值。操作系統
event.wait(): 當event對象內部的信號標識設置爲「假」(False),全部等待這個event對象的線程將會所有被阻塞。線程
event.set():設置event對象內部的信號標識爲「真」(True),全部等待這個event對象的線程將會被喚醒,等待操做系統的調度。debug
event.clear():恢復event的狀態值爲False。
下面這個圖,能讓你清楚的明白event.wait和event.set之間的關係:
3、event使用示例。
咱們來考慮下event對象的應用場景。
假如,在一個程序中,有多個線程須要到redis中來取數據,每一個線程都要去嘗試着去鏈接redis服務器,通常狀況下,redis若是鏈接不成功,在全部線程中都須要嘗試着去從新鏈接。
若是咱們想要在程序啓動時,先要確保redis服務器工做正常,接着在讓其餘工做的線程去鏈接redis服務器,當遇到這種需求的時候,使用event對象就是一個很是不錯的選擇。
咱們可使用event機制去讓各個工做的線程去作一個簡單的通訊,主線程去鏈接redis服務器,若是鏈接到了,event對象內的信號標識改成真,其他工做的線程就會被喚醒。
示例:
#!/usr/local/bin/python2.7
# -*- coding:utf-8 -*-
import threading
import time
import logging
logging.basicConfig(level=logging.DEBUG, format='(%(threadName)-10s) %(message)s',)
def worker(event):
logging.debug("waiting for redis ready....")
event.wait()
logging.debug("redis ready,and connect to redis server and do some work %s",time.ctime())
time.sleep(1)
def main():
redis_ready = threading.Event()
t1 = threading.Thread(target=worker,args=(redis_ready,),name="threading-1")
t1.start()
t2 = threading.Thread(target=worker,args=(redis_ready,),name="threading-2")
t2.start()
logging.debug("first of all,check redis server, make sure it is ok , and then trigger the redis ready event")
time.sleep(3)
redis_ready.set()
if __name__ == '__main__':
main()
輸出結果:
(threading-1) waiting for redis ready....
(threading-2) waiting for redis ready....
(MainThread) first of all,check redis server, make sure it is ok , and then trigger the redis ready event
(threading-1) redis ready,and connect to redis server and do some work Sun May 14 09:35:46 2017
(threading-2) redis ready,and connect to redis server and do some work Sun May 14 09:35:46 2017
代碼分析:
在上面這段代碼中,一共開了三個線程,分別是threading-1和threading-2還有主線程,首先主線程運行了main()函數,產生了一個event對象,這個event對象信號的初始值是False,而後建立了threading-1和threading-2兩個線程而且運行,輸出一條日誌first of all,check redis server, make sure it is ok , and then trigger the redis ready event後sleep 3秒,此時切換到了threading-1,threading-1首先執行了worker函數,輸出一第二天志,而後執行event.wait方法,當這個event對象的信號標識爲False時,threading-1就被阻塞住了,暫時沒法執行woker函數後面的內容,當threading-1被阻塞住後,隨即切換到threading-2(爲何必定切換到threading-2呢?這是由於主線程還在sleep),threading-2此時也去執行woker函數,一樣先輸出一條日誌waiting for redis ready....,當執行到event.wait時,這個event對象的信號標誌位依舊爲False,因此執行到這裏的時候,threading-2也會被阻塞。
過了3秒後,MainThread主線程sleep結束,開始向下運行,執行了redis_ready.set()後,咱們建立的那個名爲redis_ready的event對象中的信號標誌會變爲True,當這event對象的信號標誌變爲True以後,以前阻塞的threading-1和threading-2都會被喚醒,繼續執行woker函數後面的代碼。
(threading-1) redis ready,and connect to redis server and do some work Sun May 14 09:35:46 2017
(threading-2) redis ready,and connect to redis server and do some work Sun May 14 09:35:46 2017
4、event對象的補充。
threading.Event的wait方法還接受一個超時參數,默認狀況下若是事件一致沒有發生,wait方法會一直阻塞下去,而加入這個超時參數以後,若是阻塞時間超過這個參數設定的值以後,wait方法會返回。對應於上面的應用場景,若是Redis服務器一致沒有啓動,咱們但願子線程可以打印一些日誌來不斷地提醒咱們當前沒有一個能夠鏈接的Redis服務,咱們就能夠經過設置這個超時參數來達成這樣的目的:
def worker(event):
while not event.is_set():
logging.debug('Waiting for redis ready...')
event.wait(2)
logging.debug('redis ready, and connect to redis server and do some work [%s]', time.ctime())
time.sleep(1)