django 定時(shí)任務(wù) django-crontab 的使用

1、前言

在做 django 開(kāi)發(fā)需求時(shí),多多少少都會(huì)遇到需要定時(shí)任務(wù)的功能,比如定時(shí)執(zhí)行任務(wù),檢查訂單之類的??赡苁且欢螘r(shí)間,比如每隔 10分鐘執(zhí)行一次,也可能是定點(diǎn)時(shí)間,比如 14:00 執(zhí)行,也可能是長(zhǎng)時(shí)間,比如每周幾,每個(gè)月的哪一天等。查看了一下相關(guān)資料, django 定時(shí)任務(wù) django-crontab 庫(kù)比較多教程和資料,雖然 star 數(shù)才五百,但是 API 接口比較簡(jiǎn)單,接入也很方便,功能也很全面,當(dāng)然,也存在一此無(wú)法解決的問(wèn)題,使用時(shí)需要注意的。

2、使用教程

1.安裝:

pip install django-crontab

2.添加配置到 settings.py INSTALLED_APPS

INSTALLED_APPS = (
    'django_crontab',
    ...
)

3.編寫(xiě)定時(shí)函數(shù):

定時(shí)任務(wù)可以分成兩種,一種是執(zhí)行自定義的mange.py的命令,另一種是執(zhí)行自定義函數(shù)。

在django的app中新建一個(gè)myapp/cron.py文件,把需要定時(shí)執(zhí)行的代碼放進(jìn)去

示例:

def my_scheduled_job():
  pass

4.在 settings.py 中增加CRONJOBS配置

CRONJOBS = [
    ('*/5 * * * *', 'myapp.cron.my_scheduled_job')
]

也可以定義一些關(guān)鍵字參數(shù),有2種格式:

格式1:

  • 要求:cron計(jì)時(shí)通常格式(有關(guān)更多示例,請(qǐng)參閱 Wikipediacrontab.guru
  • 要求:python模塊路徑下待執(zhí)行定時(shí)任務(wù)
  • 可選:特定于定時(shí)任務(wù)的后綴(例如,將 out/err 重定向到文件,默認(rèn)值為'')

示例:

CRONJOBS = [
    ('*/1 * * * *', 'appname.test_crontab.test','>>/home/python/test_crontab.log')

]

注意: >> 表示追加寫(xiě)入,> 表示覆蓋寫(xiě)入。

格式2:

  • 要求:cron計(jì)時(shí)通常格式
  • 要求:python模塊路徑下待執(zhí)行定時(shí)任務(wù)
  • 可選:方法的位置參數(shù)列表(默認(rèn)值:[])
  • 可選:方法的關(guān)鍵字參數(shù)的dict(默認(rèn)值:{})
  • 可選:特定于定時(shí)任務(wù)的后綴(例如,將 out/err 重定向到文件,默認(rèn)值為'')

示例:

CRONJOBS = [
    ('*/5 * * * *', 'myapp.cron.other_scheduled_job', ['arg1', 'arg2'], {'verbose': 0}),
    ('0   4 * * *', 'django.core.management.call_command', ['clearsessions']),
    
]

對(duì)于熟悉 Linux 中定時(shí)任務(wù)crontab 的同學(xué)可能對(duì)上面第一個(gè)參數(shù)的語(yǔ)法很親切。上面表示每隔1分鐘執(zhí)行一次代碼。

Linux 中的定時(shí)任務(wù)crontab的語(yǔ)法如下:

*  *  *  *  * command
分鐘(0-59) 小時(shí)(0-23) 每個(gè)月的哪一天(1-31) 月份(1-12) 周幾(0-6) shell腳本或者命令

有幾個(gè)特殊的符號(hào):

* 代表所有的取值范圍的數(shù)字
/ 代表每的意思,*/5就是每5個(gè)單位
- 代表從某個(gè)數(shù)字到某個(gè)數(shù)字
, 分開(kāi)幾個(gè)離散的數(shù)字

示例:

每?jī)蓚€(gè)小時(shí)    0 */2 * * *
晚上11點(diǎn)到早上8點(diǎn)之間每?jī)蓚€(gè)小時(shí),早上8點(diǎn)    0 23-7,8 * * *
每個(gè)月的4號(hào)和每個(gè)禮拜的禮拜一到禮拜三的早上11點(diǎn)    0 11 4 * 1-3
1月1日早上4點(diǎn)    0 4 1 1 * 
0 6 * * * commands >> /tmp/test.log # 每天早上6點(diǎn)執(zhí)行, 并將信息追加到test.log中
0 */2 * * * commands # 每隔2小時(shí)執(zhí)行一次

有興趣的小伙伴可以深入研究下 Linux 的crontab定時(shí)任務(wù)。如果不了解和不熟悉可以想看: cron語(yǔ)法格式學(xué)習(xí)

5.添加并啟動(dòng)定時(shí)任務(wù)

#添加并啟動(dòng)定時(shí)任務(wù)
python manage.py crontab add

其它命令:

#顯示當(dāng)前的定時(shí)任務(wù)
python manage.py crontab show

#刪除所有定時(shí)任務(wù)
python manage.py crontab remove

一些問(wèn)題

如果配置成這樣:

BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))

CRONJOBS = [
('0 7 * * 1-5', 'api.cron.email_to_late_docs', '>> {}'.format(BASE_DIR + '/logs/log_{:%d_%m_%Y}.log'.format(time.now()))),
('0 7 * * 1-5', 'api.cron.email_ten_days_before', '>> {}'.format(BASE_DIR + 'logs/log_{:%d_%m_%Y}.log'.format(time.now())))
]

上述代碼的目的是,希望對(duì)任務(wù)的輸出和錯(cuò)誤日志,進(jìn)行 log_ddmmYY 格式進(jìn)行文件每天分開(kāi)保存。

但是這個(gè)任務(wù),在第二天時(shí),就不會(huì)在執(zhí)行啦??!

因?yàn)椋?code>CRONJOBS 生成任務(wù)時(shí),會(huì)生成對(duì)應(yīng)的哈希值(hashes),標(biāo)識(shí)每個(gè)任務(wù)。所以,當(dāng)文件名變更時(shí),CRONJOBS 中的值每天都在變化,導(dǎo)致不同的定時(shí)任務(wù)哈希值(hashes)。

針對(duì)這種情況,解決方法是,日志文件名稱固定,然后創(chuàng)建一個(gè)任務(wù),用來(lái)每天把日志文件重命名(move)成想要的格式名稱,這樣就可以啦!

總結(jié)

通過(guò)這個(gè)需求,可以看到很多知識(shí)點(diǎn)其實(shí)是串聯(lián)起來(lái)的,從pythondjangoLinuxcrontab,所以,學(xué)習(xí)無(wú)止境,知識(shí)學(xué)習(xí)只會(huì)越來(lái)越多,如果你提前掌握了某些知識(shí),那么學(xué)習(xí)新(舊)知識(shí)的成本就會(huì)降低很多,或者理解成本,比如你學(xué)習(xí)了 Linux, 了解過(guò) cron ,那么對(duì)于學(xué)習(xí)這個(gè) django 的定時(shí)任務(wù)會(huì)輕松很多!永遠(yuǎn)不要認(rèn)為有些知識(shí)你永遠(yuǎn)用不上,所以現(xiàn)在就不學(xué),可能現(xiàn)在的永遠(yuǎn)距離已經(jīng)很短啦!加油~

參考


  • 如有疑問(wèn),歡迎在評(píng)論區(qū)一起討論!
  • 如有不正確的地方,歡迎指導(dǎo)!


注:本文首發(fā)于 iHTCboy's blog,如若轉(zhuǎn)載,請(qǐng)注來(lái)源

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

友情鏈接更多精彩內(nèi)容