Django 如何使用 Celery 完成异步任务或定时任务

以前版本的 Celery 需要一个单独的库(django-celery)才能与 Django 一起工作, 但从 Celery 3.1 开始,情况便不再如此,我们可以直接通过 Celery 库来完成在 Django 中的任务。

安装 Redis 服务端

以 Docker 安装为例,安装一个密码为 mypassword 的 Redis 服务端

docker run -itd --name redis -p 127.0.0.1:6379:6379 redis:alpine redis-server --requirepass mypassword

在 Python 中安装 Celery 和 Redis

pip install celery redis

在 Django 项目中添加 Celery 配置

在 Django 项目中创建一个 celery.py 文件,并配置 Celery 应用程序。这个文件应该与 settings.py 文件位于同一目录下:

import os

from celery import Celery

# 设置 Django 的默认环境变量
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'myproject.settings')

app = Celery('myproject')

# 使用 Django 的 settings.py 文件配置 Celery
app.config_from_object('django.conf:settings', namespace='CELERY')

# 从所有已安装的应用中自动发现并加载任务模块
app.autodiscover_tasks()

然后在 settings.py 文件中添加配置:

# 使用 Redis 作为消息代理(broker)来传递任务消息,连接地址为 localhost:6379/0,并提供密码 mypassword 进行身份验证。
CELERY_BROKER_URL = 'redis://:mypassword@localhost:6379/0'

# 使用 Redis 作为结果存储后端,连接地址同上,使用相同的密码进行身份验证。
CELERY_RESULT_BACKEND = 'redis://:mypassword@localhost:6379/0'

# 指定发送到代理(broker)的任务消息序列化格式为 JSON 格式。
CELERY_TASK_SERIALIZER = 'json'

# 指定从结果后端获取的结果序列化格式为 JSON 格式。
CELERY_RESULT_SERIALIZER = 'json'

# 指定支持接收的内容类型为 JSON 格式。
CELERY_ACCEPT_CONTENT = ['json']

# 将时区设置为亚洲/上海时区。
CELERY_TIMEZONE = 'Asia/Shanghai'

# 启用 UTC 时间。
CELERY_ENABLE_UTC = True

在 Django 应用程序中创建一个 tasks.py 文件,并编写要运行的任务函数。例如,此处我们将编写一个名为 send_email() 的任务,来定期发送电子邮件:

from django.core.mail import send_mail
from celery import shared_task

@shared_task
def send_email():
    # 发送电子邮件的代码
    pass

如果想要实现异步任务的功能,在 Django 项目中的任何位置调用任务函数即可。例如,在 views.py 文件中,我们可以从视图函数中启动任务,如下所示:

from myapp.tasks import send_email

def my_view(request):
    send_email.delay()
    return HttpResponse('任务已经在后台执行。')

如果想要实现定时任务的功能,可以在 Celery 的配置文件中设置定时任务的调度方式。例如,要每小时运行一次 send_email() 任务,我们可以添加以下代码:

from celery.task.schedules import crontab

app.conf.beat_schedule = {
    'send-email-every-hour': {
        'task': 'myapp.tasks.send_email',
        'schedule': crontab(minute=0, hour='*/1'),
    },
}

定时任务的具体写法可以参考官方文档:https://docs.celeryq.dev/en/stable/userguide/periodic-tasks.html?highlight=crontab

运行 Celery-worker 与 Celery-beat

Celery是一个分布式任务队列,由三个主要组件组成:Celery worker、Celery beat 和消息代理(例如 Redis 或 RabbitMQ)。这些组件一起协作,让开发者能够轻松地执行异步任务和定时任务。

Celery worker:负责接收任务请求并执行任务。当您在 Django 应用程序中调用 apply_async 方法时,任务将被发送到 Celery worker,然后由 worker 执行。

Celery beat:负责调度定时任务。它会根据定义的规则定期触发任务,并将其发送到 Celery worker 处理。

所以,对于需要运行定时任务的情况,我们需要同时启动 Celery worker 和 Celery beat 进程来确保所有任务都可以被正确地处理和执行。

如果只需要使用 Celery 来执行异步任务,那么只需启动 Celery worker 即可。但如果需要周期性地执行任务,那么需要启动 Celery beat 来帮助完成调度这些任务。

# 运行 worker 与 beat
celery -A proj worker --loglevel=info --detach --pidfile=worker.pid --logfile=./logs/worker.log
celery -A proj beat --loglevel=info --detach --pidfile=beat.pid --logfile=./logs/beat.log
  • -A proj:指定 Celery 应用程序所在的模块或包,这里假设其名为 proj。
  • worker 或 beat:启动的进程名称,分别对应 worker 和 beat 两种类型的 Celery 进程。
  • --loglevel=info:设置日志级别为 info,即只记录 info 级别及以上的日志信息。
  • --detach:以守护进程(daemonized)方式启动 Celery 进程,使其在后台运行。
  • --pidfile=worker.pid 或 --pidfile=beat.pid:将进程 ID(PID)写入指定的 PID 文件,方便后续管理和监控。
  • --logfile=./logs/worker.log 或 --logfile=./logs/beat.log:指定日志文件路径,所有日志信息都会输出到该文件中。

随后我们设定的定时任务便会按规则执行,可以通过指定的日志文件查看执行结果。当我们需要停止 Celery worker 与 Celery beat 时,可以执行以下操作:

kill -TERM $(cat worker.pid)
kill -TERM $(cat beat.pid)

参考

[1] [Using Celery with Django]: https://docs.celeryq.dev/en/stable/django/first-steps-with-django.html#using-celery-with-django
[2] [Periodic Tasks]: https://docs.celeryq.dev/en/stable/userguide/periodic-tasks.html?highlight=crontab

原文链接:https://www.cnblogs.com/desireroot7/p/17352598.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:Django 如何使用 Celery 完成异步任务或定时任务 - Python技术站

(0)
上一篇 2023年4月25日
下一篇 2023年4月25日

相关文章

  • Python中输入和输出(打印)数据实例方法

    作为Python程序员,输入和输出数据通常是我们的必备技能之一,Python提供了多种实例方法可以很方便地实现这个目标,下面我们就来详细讲解一下这些实例方法。 输入数据的实例方法 Python中实现输入数据的方法通常使用input()函数,它的形式如下: input([prompt]) 其中prompt是一个字符串,用于表示输入时的提示信息。如果没有指定pr…

    python 2023年5月19日
    00
  • python多线程互斥锁与死锁问题详解

    Python多线程互斥锁与死锁问题详解 多线程是Python中很重要的特性,但是在多线程编程中,会遇到一些问题,比如互斥锁(mutex)和死锁(deadlock)问题。本文将详细探讨多线程中的互斥锁和死锁问题,并提供两个示例说明。 什么是互斥锁? 在多线程编程中,多个线程同时访问共享资源时,会发生资源冲突的问题,导致程序出现错误。为了避免这种情况,可以使用互…

    python 2023年5月19日
    00
  • Python自动爬取图片并保存实例代码

    Python自动爬取图片并保存实例代码 本攻略将介绍如何使用Python自动爬取图片并保存到本地。我们将使用Python的requests库和BeautifulSoup库来获取和解析网页内容,使用os库和urllib库来创建和保存图片文件。 获取图片链接 我们可以使用Python的requests库和BeautifulSoup库来获取图片链接。以下是一个示例…

    python 2023年5月15日
    00
  • python实现人机对战的井字棋游戏

    Python实现人机对战的井字棋游戏 概述 本文将详细讲解如何使用Python语言实现人机对战的井字棋游戏。井字棋游戏是一款简单的棋类游戏,由于其简单易懂、规则简单,非常适合用来练手。在实现本游戏时,我们将使用Python的面向对象编程思想,通过类的定义和方法的调用实现游戏的逻辑。同时,我们也将使用Python的标准库Tkinter实现简单的GUI界面,让游…

    python 2023年5月23日
    00
  • Python根据成绩分析系统浅析

    下面就是“Python根据成绩分析系统浅析”的完整攻略。 系统概述 该系统是一个基于Python实现的成绩分析系统,旨在通过分析学生的各项成绩数据,为学生提供更好的学习监督和指导,教师提供更好的学科教学指导。 系统结构 系统分为两部分:数据爬取和分析计算。 数据爬取 数据爬取部分负责从学校教务系统爬取学生的成绩数据,并存储到本地或者云端数据库中,以便后续的分…

    python 2023年5月30日
    00
  • Python中实现switch功能实例解析

    下面是关于“Python中实现switch功能实例解析”的完整攻略。 概述 在Python中,没有类似于C++或Java中的switch-case语句来实现多个分支的条件判断。但是,我们可以使用字典(dict)和函数来实现类似于switch-case的功能。下面就让我们一步步来看如何实现。 方法1:使用字典实现 使用字典实现switch-case语句的思路是…

    python 2023年5月19日
    00
  • python获得一个月有多少天的方法

    想要获得一个月有多少天,可以借助datetime库中的date类和calendar库中的monthrange函数。 首先导入需要的库: from datetime import date import calendar 然后定义一个日期变量,指定一个月和年份: year = 2022 month = 3 day = 1 my_date = date(year…

    python 2023年6月2日
    00
  • Python的时间模块datetime详解

    Python的时间模块datetime详解 简介 在Python中,datetime是一个重要的时间处理模块,它可以处理日期、时间、时间差等内容,是处理时间和日期相关操作的首选模块。本文将对datetime模块做一个详细的介绍。 datetime模块的基本用法 datetime模块提供了三个类:datetime、date和time。其中datetime是使用…

    python 2023年6月2日
    00
合作推广
合作推广
分享本页
返回顶部