Django Celery异步任务队列的实现


Posted in Python onJuly 24, 2019

背景

在开发中,我们常常会遇到一些耗时任务,举个例子:

上传并解析一个 1w 条数据的 Excel 文件,最后持久化至数据库。

在我的程序中,这个任务耗时大约 6s,对于用户来说,6s 的等待已经是个灾难了。

比较好的处理方式是:

  1. 接收这个任务的请求
  2. 将这个任务添加到队列中
  3. 立即返回「操作成功,正在后台处理」的字样
  4. 后台消费这个队列,执行这个任务

我们按照这个思路,借助 Celery 进行实现。

实现

本文所使用的环境如下:

  • Python 3.6.7
  • RabbitMQ 3.8
  • Celery 4.3

使用 Docker 安装 RabbitMQ

Celery 依赖一个消息后端,可选方案有 RabbitMQ, Redis 等,本文选用 RabbitMQ 。

同时为了安装方便,RabbitMQ 我直接使用 Docker 安装:

docker run -d --name anno-rabbit -p 5672:5672 rabbitmq:3

启动成功后,即可通过 amqp://localhost 访问该消息队列。

安装并配置 Celery

Celery 是 Python 实现的工具,安装可以直接通过 Pip 完成:

pip install celery

同时假设当前我的项目文件夹为 proj ,项目名为 myproj ,应用名为 myapp

安装完成后,在 proj/myproj/ 路径下创建一个 celery.py 文件,用来初始化 Celery 实例:

proj/myproj/celery.py

from __future__ import absolute_import, unicode_literals
import os
from celery import Celery, platforms

# set the default Django settings module for the 'celery' program.
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'myproj.settings')

app = Celery('myproj',
       broker='amqp://localhost//',
       backend='amqp://localhost//')

# Using a string here means the worker don't have to serialize
# the configuration object to child processes.s
# - namespace='CELERY' means all celery-related configuration keys
#  should have a `CELERY_` prefix.
app.config_from_object('django.conf:settings', namespace='CELERY')

# Load task modules from all registered Django app configs.
app.autodiscover_tasks()

然后在 proj/myproj/__init__.py 中添加对 Celery 对象的引用,确保 Django 启动后能够初始化 Celery:

proj/myproj/__init__.py

from __future__ import absolute_import, unicode_literals

# This will make sure the app is always imported when
# Django starts so that shared_task will use this app.
from .celery import app as celery_app

__all__ = ('celery_app',)

无其他特殊配置的话,Celery 的基本配置就是这些。

编写一个耗时任务

为了模拟一个耗时任务,我们直接创建一个方法,使其「睡」10s ,并将其设置为 Celery 的任务:

proj/myapp/tasks.py

import time
from myproj.celery import app as celery_app

@celery_app.task
def waste_time():
  time.sleep(10)
  return "Run function 'waste_time' finished."

启动 Celery Worker

Celery 配置完成,并且任务创建成功后,我们以异步任务的模式启动 Celery :

celery -A myproj worker -l info

注意到我强调了异步模式,是因为 Celery 除了支持异步任务,还支持定时任务,因此启动时候要指明。

同时要注意,Celery 一旦启动,对 Task(此处为 waste_time) 的修改必须重启 Celery 才会生效。

任务调用

在请求处理的逻辑代码中,调用上面创建好的任务:

proj/myapp/views.py

from django.http import JsonResponse
from django.views.decorators.http import require_http_methods
from .tasks import waste_time

@require_http_methods(["POST"])
def upload_files(request):
  waste_time.delay()
  # Status code 202: Accepted, 表示异步任务已接受,可能还在处理中
  return JsonResponse({"results": "操作成功,正在上传,请稍候..."}, status=202)

调用 waste_time.delay() 方法后, waste_time 会被加入到任务队列中,等待空闲的 Celery Worker 调用。

效果

当我们发送请求时,这个接口会直接返回 {"results": "操作成功,正在上传,请稍候..."} 的响应内容而非卡住十秒,用户体验要好许多。

总结

用 Celery 处理这种异步任务是 Python 常用的方法,虽然实际执行成功耗时不变甚至有所增加(如 Worker 繁忙导致处理滞后),但是对于用户体验来说更容易接受,点击上传大文件后可以继续处理其他事务,而不需要在页面等待。
Celery 还有更多用法本文未介绍到,其文档已经非常详尽,有需要可直接参考。

参考

http://docs.celeryproject.org/en/latest/django/first-steps-with-django.html

https://hub.docker.com/_/rabbitmq

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python读取csv文件示例(python操作csv)
Mar 11 Python
python集合类型用法分析
Apr 08 Python
在Python中使用第三方模块的教程
Apr 27 Python
python切片及sys.argv[]用法详解
May 25 Python
python使用webdriver爬取微信公众号
Aug 31 Python
Python多线程原理与用法实例剖析
Jan 22 Python
python实现大转盘抽奖效果
Jan 22 Python
python3使用腾讯企业邮箱发送邮件的实例
Jun 28 Python
python yield关键词案例测试
Oct 15 Python
python实现连续变量最优分箱详解--CART算法
Nov 22 Python
Python fileinput模块如何逐行读取多个文件
Oct 05 Python
Python内置类型集合set和frozenset的使用详解
Apr 26 Python
python如何统计代码运行的时长
Jul 24 #Python
Django时区详解
Jul 24 #Python
详解Django定时任务模块设计与实践
Jul 24 #Python
Python3中urlencode和urldecode的用法详解
Jul 23 #Python
对python3中的RE(正则表达式)-详细总结
Jul 23 #Python
python正则表达式匹配不包含某几个字符的字符串方法
Jul 23 #Python
python使用百度文字识别功能方法详解
Jul 23 #Python
You might like
php计划任务之ignore_user_abort函数实现方法
2015/01/08 PHP
PHP实现加密文本文件并限制特定页面的存取的效果
2016/10/21 PHP
php实现基于pdo的事务处理方法示例
2017/07/21 PHP
php实现微信支付之企业付款
2018/05/30 PHP
jquery 分页控件实现代码
2009/11/30 Javascript
mysql输出数据赋给js变量报unterminated string literal错误原因
2010/05/22 Javascript
Javascript的一种模块模式
2010/09/08 Javascript
js判断一个元素是否为另一个元素的子元素的代码
2012/03/21 Javascript
js实现广告漂浮效果的小例子
2013/07/02 Javascript
从QQ网站中提取的纯JS省市区三级联动菜单
2013/12/25 Javascript
js二维数组定义和初始化的三种方法总结
2014/03/03 Javascript
JS获取及设置TextArea或input文本框选择文本位置的方法
2015/03/24 Javascript
如何根据百度地图计算出两地之间的驾驶距离(两种语言js和C#)
2015/10/29 Javascript
javascript设置和获取cookie的方法实例详解
2016/01/05 Javascript
微信小程序 教程之模板
2016/10/18 Javascript
NodeJS整合银联网关支付(DEMO)
2016/11/09 NodeJs
关于jQuery中fade(),show()起始位置的一点小发现
2017/04/25 jQuery
JavaScript中offsetWidth的bug及解决方法
2017/05/17 Javascript
angular-ngSanitize模块-$sanitize服务详解
2017/06/13 Javascript
jQuery:unbind方法的使用详解
2017/08/14 jQuery
(模仿京东用户注册)用JQuery实现简单表单验证,初学者必看
2018/01/08 jQuery
python3 实现的人人影视网站自动签到
2016/06/19 Python
python发送邮件功能实现代码
2016/07/15 Python
Python编程实现删除VC临时文件及Debug目录的方法
2017/03/22 Python
python机器学习实战之树回归详解
2017/12/20 Python
python3解析库BeautifulSoup4的安装配置与基本用法
2018/06/26 Python
Python logging设置和logger解析
2019/08/28 Python
python中调试或排错的五种方法示例
2019/09/12 Python
python判断单向链表是否包括环,若包含则计算环入口的节点实例分析
2019/10/23 Python
Qoo10台湾站:亚洲领先的在线市场
2018/05/15 全球购物
牵手50香港:专为黄金岁月的单身人士而设的交友网站
2020/08/14 全球购物
个人师德师风自我剖析材料
2014/09/29 职场文书
工会2014法制宣传日活动总结
2014/11/01 职场文书
高温慰问简报
2015/07/21 职场文书
大学入学感言
2015/08/01 职场文书
导游词之开封禹王台风景区
2019/12/02 职场文书