爬虫架构|Celery+RabbitMQ快速入门(三)

在之前两章节中,简单介绍了Celery+RabbitMQ,以及它们之间的协作过程(见文章爬虫架构|Celery+RabbitMQ快速入门(一)爬虫架构|Celery+RabbitMQ快速入门(二))。
我们一直在说“Celery是一个基于Python开发的分布式异步消息队列,可以轻松实现任务的异步处理。它的基本工作就是管理分配任务到不同的服务器,并且取得结果”,可以得知,我们之所以使用它是看中了它的分布式,我们使用的场景也是用它做分布式爬虫架构(为什么不选用scrapy-redis?以及它们之间的区别,下次再讲)。
对于一个分布式爬虫来说,有两个最基本的问题需要解决。

  • 分配爬取任务:为每个爬虫分配不重复的爬取任务。
  • 汇总爬取结果:将所有爬虫爬取到的数据汇总到一处。

接下来从Celery+RabbitMQ组合中去看它们是如何解决这两个问题的。为了下面便于讲解,先把Celery+Broker(RabbitMQ)的工作流程图记录如下图3-1所示。

图3-1 Celery+Broker工作流程

一、分配爬取任务
上面说到,在分配爬取任务时需要解决的问题是为每个爬虫分配不重复的爬取任务,Celery+RabbitMQ给出的解决方案是把所有的爬取任务放在一起,并且在获取任务时进行去重。
1.1、爬取任务汇总一起
Celery+RabbitMQ为多个爬虫分配爬取任务的方式是:让所有爬虫(即图上3-1的worker)共享一个存在于RabbitMQ中的请求队列,用来替代各爬虫独立的请求队列,每个爬虫从请求队列中获取爬取任务进行数据采集,Celery是RabbitMQ中任务的生产者,各个爬虫(worker)是任务的消费者。
Celery通过app.task函数produce任务到RabbitMQ时可以采用独立的配置文件定义一些produce任务的方式和参数。
配置名称一般为celeryconfig.py(当然也可以使用任意的模块名),通过调用 config_from_object() 来让 Celery 实例加载配置模块:

app = Celery()
app.config_from_object(celeryconfig)

celeryconfig.py配置文件内容如下:

#!/usr/bin/env python
# -*- coding: UTF-8 -*-

from __future__ import absolute_import, unicode_literals
from kombu import Queue,Exchange
from celery.schedules import crontab

BROKER_URL='amqp://spider:****@IP:端口/yimian'

#默认celery与broker的连接池连接数
BROKER_POOL_LIMIT = 10

CELERY_ACKS_LATE = True
CELERY_IGNORE_RESULT = True
CELERY_DISABLE_RATE_LIMITS = True
BROKER_TRANSPORT_OPTIONS = {'visibility_timeout': 86400}
WORKER_MAX_MEMORY_PER_CHILD = 600
CELERYD_MAX_TASKS_PER_CHILD = 1
CELERY_TASK_SERIALIZER = 'json'
#CELERY_RESULT_SERIALIZER = 'json'
CELERY_ACCEPT_CONTENT = ['json']
CELERY_ENABLE_UTC = False
CELERY_TIMEZONE = 'Asia/Shanghai'
TIME_ZONE = 'Asia/Shanghai'

# 配置队列
CELERY_QUEUES = {
        Queue('default',Exchange('default'),routing_key='default'),
        Queue('spider_001',Exchange('spider_001'),routing_key='spider_001'),
        Queue('spider_002',Exchange('spider_002'),routing_key='spider_002'),
        Queue('spider_003',Exchange('spider_003'),routing_key='spider_003'),
}

#队列路由
CELERY_ROUTES = {
    'spider_name.tasks.daily_spider_001':{'queue':'spider_001','routing_key':'spider_001'},
    'spider_name.tasks.daily_spider_002':{'queue':'spider_002','routing_key':'spider_002'},
    'spider_name.tasks.daily_spider_003':{'queue':'spider_003','routing_key':'spider_003'}
}

在celeryconfig.py文件中,首先设置了Broker(RabbitMQ)的URL,接下来定义了三个Message Queue,并且指明了Queue对应的Exchange(当使用Redis作为Broker时,Exchange的名字必须和Queue的名字一样)以及routing_key的值。
CELERY_QUEUES中的routing_key与CELERY_ROUTES中的routing_key是一一对应的关系。
1.2、获取任务时去重
如上我们在生产任务时已经把任务分到了不同的队列中,在启动worker进行消费任务时可以使用-Q Queue_Name参数指定需要消费哪个队列中的任务。

celery -A tasks worker -Q spider_001

其中-Q参数指定了这个worker执行spider_001队列中的消息。一般情况下,会有多个worker消费一个队列中的任务。至于多个worker为什么不会出现消费同一个任务,这里是celery本身的负载均衡的机制保障了任务去重。

二、汇总爬取结果
在分布式爬虫中,各个服务器爬到的数据最终要汇总到一处,比如到MySQL数据库。

三、Celery后续
Celery由5个主要组件组成:

  • producer: 任务发布者, 通过调用API向celery发布任务的程序
  • celery beat: 任务调度, 根据配置文件发布定时任务
  • worker: 实际执行任务的程序
  • broker: 接受任务消息,存入队列再按顺序分发给worker执行
  • backend: 存储结果的服务器

还剩下celery beat和backend没有讲解,后面会有一篇爬虫架构|Celery+RabbitMQ快速入门(四),将汇总一、二、三的所有内容完整地整理一下。
所以,
一、二、三没有看懂没有关系,等待我的第四篇文章。
一、二、三没有看懂没有关系,等待我的第四篇文章。
一、二、三没有看懂没有关系,等待我的第四篇文章。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,736评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,167评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,442评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,902评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,302评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,573评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,847评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,562评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,260评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,531评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,021评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,367评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,016评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,068评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,827评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,610评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,514评论 2 269

推荐阅读更多精彩内容