新聞中心
有時候,我們想實(shí)現(xiàn)一個非常簡單的定時功能,例如讓一個程序每天早上8點(diǎn)調(diào)用某個函數(shù)。但我們又不想安裝任何第三方庫,也不會使用 crontab 或者任務(wù)計(jì)劃功能,就想使用純 Python 來實(shí)現(xiàn)。

可能有同學(xué)會這樣寫代碼:
- import time
- import datetime
- def run():
- print('我是需要被每天調(diào)用的函數(shù)')
- def schedule():
- target_time = datetime.time(8, 0, 0)
- today = datetime.date.today()
- target_date = today + datetime.timedelta(days=1)
- target_datetime = datetime.datetime.combine(target_date, target_time)
- now = datetime.datetime.now()
- delta = (target_datetime - now).total_seconds()
- time.sleep(delta)
- run()
- while True:
- time.sleep(24 * 3600)
- run()
- if __name__ == '__main__':
- schedule()
這段程序,首先計(jì)算出現(xiàn)在距離明天早上8點(diǎn)相差的秒數(shù)。睡這么多秒以后,第一次運(yùn)行目標(biāo)函數(shù)。然后進(jìn)入一個死循環(huán),每隔86400秒,程序調(diào)用一次 run 函數(shù)。
這個程序初看起來,似乎沒有什么問題。但如果你每天觀察它的運(yùn)行時間,你會發(fā)現(xiàn)隨著時間的推移,時間會越來越不準(zhǔn)確。
這是因?yàn)?,run 函數(shù)不是一瞬間就運(yùn)行完成的。它運(yùn)行也會消耗時間。假設(shè)程序第一次運(yùn)行 run 函數(shù)的時候,確實(shí)剛剛好是8:00,run 函數(shù)運(yùn)行了2秒。那么,程序睡眠86400秒以后,時間實(shí)際上是8:00:02.從第二天開始,每天晚2秒鐘。一個月就會晚一分鐘。
但實(shí)際上,我們?nèi)绻冻鲆稽c(diǎn)點(diǎn)微不足道的代價,我們就可以防止這種誤差的發(fā)生,并且程序代碼會變得更簡單:
- import time
- import datetime
- def run():
- print('我是需要被每天調(diào)用的函數(shù)')
- def schedule():
- last_run = None
- while True:
- now = datetime.datetime.now()
- if now.strftime('%H:%M') == '08:00' and last_run != now.date():
- run()
- last_run = now.date()
- time.sleep(1)
- if __name__ == '__main__':
- schedule()
程序在一個死循環(huán)中,每秒做一次檢查,如果當(dāng)前的時分正好是08:00,并且上一次運(yùn)行不是今天,那么就調(diào)用 run 函數(shù),并把上一次運(yùn)行的時間設(shè)置為今天。否則,就睡眠1秒鐘。
這樣做,相當(dāng)于每秒都會校對時間,從而避免了長時間運(yùn)行導(dǎo)致的時間誤差。雖然看起來這個死循環(huán)會非常消耗 CPU,但只要你算一下,實(shí)際上它只不過每天循環(huán)86400次而已。這個次數(shù)并不多。
但無論如何,專業(yè)的事情應(yīng)該交由專業(yè)的工具來做。time.sleep用來設(shè)置周期性的時間間隔可以,但它實(shí)際上不適合用來做定時任務(wù)。
因?yàn)橐粋€支持定時任務(wù)的庫,例如 Python 的schedule或者APScheduler,他們在確保定時時間準(zhǔn)確上,做了很多工作。還有一些庫甚至用到了時間輪這樣的數(shù)據(jù)結(jié)構(gòu)來確保時間的準(zhǔn)確性。這不是我們簡單用兩三行 Python 代碼就能完成的。
總結(jié)
如果能用 crontab 或者任務(wù)計(jì)劃,那么這是最優(yōu)選擇。其次,使用 Python 專用的定時模塊。最次,才是使用 time.sleep 來實(shí)現(xiàn)。如果不得不用 time.sleep,那么應(yīng)該盡量縮短檢查的間隔,避免長時間睡眠。
本文轉(zhuǎn)載自微信公眾號「未聞Code」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請聯(lián)系未聞Code公眾號。
本文題目:為什么不建議使用Time.Sleep實(shí)現(xiàn)定時功能?
文章轉(zhuǎn)載:http://www.5511xx.com/article/djdissj.html


咨詢
建站咨詢
