Celery周期性定时任务-periodic_task

周期性定时任务

说明:在Django中使用celery, 并结合django-celery模块(省略安装)。需要在配置文件中注册:

import djcelery
INSTALLED_APPS += ('djcelery',)
CELERY_ENABLE_UTC = False
CELERYBEAT_SCHEDULER = "djcelery.schedulers.DatabaseScheduler"  # 基于数据库模型调度

基本使用

  • django中使用,celery_tasks.py:

    from celery.schedules import crontab
    from celery.task import periodic_task
    
    # 每天9点执行
    @periodic_task(run_every=crontab(minute='0', hour='9'))
    def job():
        # 业务逻辑
        print('周期定时任务!')
    

crontab表达式

详见官网: https://docs.celeryproject.org/en/stable/userguide/periodic-tasks.html

例子 意义
crontab() 每分钟执行一次。
crontab(minute=0, hour=0) 每天午夜执行。
crontab(minute=0, hour='*/3') 每三个小时执行一次:午夜、凌晨 3 点、早上 6 点、早上 9 点、中午、下午 3 点、下午 6 点、晚上 9 点。
crontab(minute=0,``hour='0,3,6,9,12,15,18,21') 和以前一样。
crontab(minute='*/15') 每 15 分钟执行一次。
crontab(day_of_week='sunday') 在星期日每分钟 (!) 执行一次。
crontab(minute='*',``hour='*', day_of_week='sun') 和以前一样。
crontab(minute='*/10',``hour='3,17,22', day_of_week='thu,fri') 每十分钟执行一次,但仅限于周四或周五的凌晨 3-4 点、下午 5-6 点和晚上 10-11 点。
crontab(minute=0, hour='*/2,*/3') 每隔偶数小时执行一次,并且每小时可被 3 整除。这意味着:除了:1am、5am、7am、11am、1pm、5pm、7pm、11pm之外的每个小时
crontab(minute=0, hour='*/5') 执行可被 5 整除的小时。这意味着它在下午 3 点而非下午 5 点触发(因为下午 3 点等于 24 小时时钟值“15”,可被 5 整除)。
crontab(minute=0, hour='*/3,8-17') 每小时执行一次,可被 3 整除,并在办公时间(上午 8 点至下午 5 点)每小时执行一次。
crontab(0, 0, day_of_month='2') 在每个月的第二天执行。
crontab(0, 0,``day_of_month='2-30/2') 在每个偶数日执行。
crontab(0, 0,``day_of_month='1-7,15-21') 在每月的第一周和第三周执行。
crontab(0, 0, day_of_month='11',``month_of_year='5') 每年5月11日执行。
crontab(0, 0,``month_of_year='*/3') 在每个季度的第一个月每天执行。
                       |

自定义周期任务

celery_tasks.py文件:定义任务加task()

from celery.task import task
@task()
def send_msg(oa_list, content, id, username, msg):
    # 逻辑
    pass

views.py写自定义方法:

import datetime
import json
from djcelery.models import PeriodicTask, CrontabSchedule

def create_task(request):
    task_name = "test"	# 自定义,不能重复,唯一
    task = 'app01.celery_tasks.send_msg'  # 任务的注册路径
    task_args = [oa_list, content]
    task_kwargs = {'id': id, 'username': username, 'msg': msg}
    # 定时规则
    crontab_time = {
    	'minute': '*/1',
    	'hour': '*',
        'day_of_week': '*'
    	'day_of_month': '*',
    	'month_of_year': '*'
    }
    
    schedule = CrontabSchedule.objects.create(**crontab_time)  # 创建时间规则

    task, created = PeriodicTask.objects.get_or_create(
    	name=task_name,		# 名称保持唯一
    	task=task,	
    	crontab=schedule,
    	enabled=True,	# 是否开启
    	args=json.dumps(task_args)
    	kwargs=json.dumps(task_kwargs),
    	expires=datetime.datetime.now()+datetime.timedelta(days=1)	# 过期时间
    )
    if created:
        return HttpResponse("创建成功")
    else:
        return HttpResponse("创建失败")

使用程序停止和删除周期定时任务

周期定时任务删除,直接通过自定义任务的name字段删除

一般是针对自定义周期任务的删除,写在celery_tasks.py中的任务这样删除后,如果重启celery,还会注册进去,因为程序启动会注册所有app下的任务。

from djcelery.schedulers import ModelEntry, DatabaseScheduler

def delete_celery_task(task_name):
	DatabaseScheduler.delete_task(task_name)

停止运行中的周期定时任务

通过PeriodicTask表中的enabled字段来控制周期任务的启停(不用重启celery)。

from djcelery.models import PeriodicTask, CrontabSchedule

def celery_enabled(task_name):
    """
    停止周期任务
    :param task_name: 周期任务名
    """
    
    task = PeriodicTask.objects.filter(name=task_name).first()
    task.enabled=False	# False停止, True开启
    task.save()    # 不要用update,任务不会停止
posted @ 2021-09-30 15:58  SensorError  阅读(2465)  评论(1编辑  收藏  举报