爬虫 之 scrapy框架
浏览目录
介绍
Scrapy一个开源和协作的框架,其最初是为了页面抓取 (更确切来说, 网络抓取 )所设计的,使用它可以以快速、简单、可扩展的方式从网站中提取所需的数据。但目前Scrapy的用途十分广泛,可用于如数据挖掘、监测和自动化测试等领域,也可以应用在获取API所返回的数据(例如 Amazon Associates Web Services ) 或者通用的网络爬虫。
Scrapy 是基于twisted框架开发而来,twisted是一个流行的事件驱动的python网络框架。因此Scrapy使用了一种非阻塞(又名异步)的代码来实现并发。整体架构大致如下
在Scrapy的数据流是由执行引擎控制,具体流程如下:
1、spiders产生request请求,将请求交给引擎
2、引擎(EGINE)吧刚刚处理好的请求交给了调度器,以一个队列或者堆栈的形式吧这些请求保存起来,调度一个出来再传给引擎
3、调度器(SCHEDULER)返回给引擎一个要爬取的url
4、引擎把调度好的请求发送给download,通过中间件发送(这个中间件至少有 两个方法,一个请求的,一个返回的),
5、一旦完成下载就返回一个response,通过下载器中间件,返回给引擎,引擎把response 对象传给下载器中间件,最后到达引擎
6、引擎从下载器中收到response对象,从下载器中间件传给了spiders(spiders里面做两件事,1、产生request请求,2、为request请求绑定一个回调函数),spiders只负责解析爬取的任务。不做存储,
7、解析完成之后返回一个解析之后的结果items对象及(跟进的)新的Request给引擎
就被ITEM PIPELUMES处理了
8、引擎将(Spider返回的)爬取到的Item给Item Pipeline,存入数据库,持久化,如果数据不对,可重新封装成一个request请求,传给调度器
9、(从第二步)重复直到调度器中没有更多地request,引擎关闭该网站.
scrapy框架分为七大部分核心的组件
1、引擎(EGINE)
引擎负责控制系统所有组件之间的数据流,并在某些动作发生时触发事件。有关详细信息,请参见上面的数据流部分。
2、调度器(SCHEDULER)
用来接受引擎发过来的请求, 压入队列中, 并在引擎再次请求的时候返回. 可以想像成一个URL的优先级队列, 由它来决定下一个要抓取的网址是什么, 同时去除重复的网址
3、下载器(DOWLOADER)
用于下载网页内容, 并将网页内容返回给EGINE,下载器是建立在twisted这个高效的异步模型上的
4、爬虫(SPIDERS)
SPIDERS是开发人员自定义的类,用来解析responses,并且提取items,或者发送新的请求
5、项目管道(ITEM PIPLINES)
在items被提取后负责处理它们,主要包括清理、验证、持久化(比如存到数据库)等操作
6、下载器中间件(Downloader Middlewares)
下载器中间件是在引擎及下载器之间的特定钩子(specific hook),处理Downloader传递给引擎的response。 其提供了一个简便的机制,通过插入自定义代码来扩展Scrapy功能。更多内容请看 下载器中间件(Downloader Middleware) 。
7、爬虫中间件(Spider Middlewares)
Spider中间件是在引擎及Spider之间的特定钩子(specific hook),处理spider的输入(response)和输出(items及requests)。 其提供了一个简便的机制,通过插入自定义代码来扩展Scrapy功能。更多内容请看 Spider中间件(Middleware) 。
安装
#Windows平台 1、pip3 install wheel #安装后,便支持通过wheel文件安装软件,wheel文件官网:https://www.lfd.uci.edu/~gohlke/pythonlibs 3、pip3 install lxml 4、pip3 install pyopenssl 5、下载并安装pywin32:https://sourceforge.net/projects/pywin32/files/pywin32/ 6、下载twisted的wheel文件:http://www.lfd.uci.edu/~gohlke/pythonlibs/#twisted 7、执行pip3 install 下载目录\Twisted-17.9.0-cp36-cp36m-win_amd64.whl 8、pip3 install scrapy #Linux平台 1、pip3 install scrapy
项目结构
project_name/ scrapy.cfg project_name/ __init__.py items.py pipelines.py settings.py spiders/ __init__.py 爬虫1.py 爬虫2.py
注意:一般创建爬虫文件时,以网站域名命名
默认只能在cmd中执行爬虫,如果想在pycharm中执行需要做:
#在项目目录下新建:entrypoint.py from scrapy.cmdline import execute # execute(['scrapy', 'crawl', 'amazon','--nolog']) #不要日志打印 # execute(['scrapy', 'crawl', 'amazon']) #我们可能需要在命令行为爬虫程序传递参数,就用下面这样的命令 #acrapy crawl amzaon -a keyword=iphone8 execute(['scrapy', 'crawl', 'amazon1','-a','keyword=iphone8','--nolog']) #不要日志打印 # execute(['scrapy', 'crawl', 'amazon1'])
windows编码
import sys,os sys.stdout=io.TextIOWrapper(sys.stdout.buffer,encoding='gb18030')
常用命令行工具
#1 查看帮助 scrapy -h scrapy <command> -h #2 有两种命令:其中Project-only必须切到项目文件夹下才能执行,而Global的命令则不需要 Global commands: startproject #创建项目 genspider #创建爬虫程序 settings #如果是在项目目录下,则得到的是该项目的配置 runspider #运行一个独立的python文件,不必创建项目 shell #scrapy shell url地址 在交互式调试,如选择器规则正确与否 fetch #独立于程单纯地爬取一个页面,可以拿到请求头 view #下载完毕后直接弹出浏览器,以此可以分辨出哪些数据是ajax请求 version #scrapy version 查看scrapy的版本,scrapy version -v查看scrapy依赖库的版本 Project-only commands: crawl #运行爬虫,必须创建项目才行,确保配置文件中ROBOTSTXT_OBEY = False check #检测项目中有无语法错误 list #列出项目中所包含的爬虫名 edit #编辑器,一般不用 parse #scrapy parse url地址 --callback 回调函数 #以此可以验证我们的回调函数是否正确 bench #scrapy bentch压力测试 #3 官网链接 https://docs.scrapy.org/en/latest/topics/commands.html
1 全局命令:所有文件夹都使用的命令,可以不依赖与项目文件也可以执行 2 项目的文件夹下执行的命令 3 1、scrapy startproject Myproject #创建项目 4 cd Myproject 5 2、scrapy genspider baidu www.baidu.com #创建爬虫程序,baidu是爬虫名,定位爬虫的名字 6 #写完域名以后默认会有一个url, 7 3、scrapy settings --get BOT_NAME #获取配置文件 8 #全局:4、scrapy runspider budui.py 9 5、scrapy runspider AMAZON\spiders\amazon.py #执行爬虫程序 10 在项目下:scrapy crawl amazon #指定爬虫名,定位爬虫程序来运行程序 11 #robots.txt 反爬协议:在目标站点建一个文件,里面规定了哪些能爬,哪些不能爬 12 # 有的国家觉得是合法的,有的是不合法的,这就产生了反爬协议 13 # 默认是ROBOTSTXT_OBEY = True 14 # 修改为ROBOTSTXT_OBEY = False #默认不遵循反扒协议 15 6、scrapy shell https://www.baidu.com #直接超目标站点发请求 16 response 17 response.status 18 response.body 19 view(response) 20 7、scrapy view https://www.taobao.com #如果页面显示内容不全,不全的内容则是ajax请求实现的,以此快速定位问题 21 8、scrapy version #查看版本 22 9、scrapy version -v #查看scrapy依赖库锁依赖的版本 23 10、scrapy fetch --nolog http://www.logou.com #获取响应的内容 24 11、scrapy fetch --nolog --headers http://www.logou.com #获取响应的请求头 25 (venv3_spider) E:\twisted\scrapy框架\AMAZON>scrapy fetch --nolog --headers http://www.logou.com 26 > Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8 27 > Accept-Language: en 28 > User-Agent: Scrapy/1.5.0 (+https://scrapy.org) 29 > Accept-Encoding: gzip,deflate 30 > 31 < Content-Type: text/html; charset=UTF-8 32 < Date: Tue, 23 Jan 2018 15:51:32 GMT 33 < Server: Apache 34 >代表请求 35 <代表返回 36 10、scrapy shell http://www.logou.com #直接朝目标站点发请求 37 11、scrapy check #检测爬虫有没有错误 38 12、scrapy list #所有的爬虫名 39 13、scrapy parse http://quotes.toscrape.com/ --callback parse #验证回调函函数是否成功执行 40 14、scrapy bench #压力测试
Spiders爬虫
介绍
1、Spiders是由一系列类(定义了一个网址或一组网址将被爬取)组成,具体包括如何执行爬取任务并且如何从页面中提取结构化的数据。
2、换句话说,Spiders是你为了一个特定的网址或一组网址自定义爬取和解析页面行为的地方
Spiders会循环做如下事情
#1、生成初始的Requests来爬取第一个URLS,并且标识一个回调函数 第一个请求定义在start_requests()方法内默认从start_urls列表中获得url地址来生成Request请求,默认的回调函数是parse方法。回调函数在下载完成返回response时自动触发 #2、在回调函数中,解析response并且返回值 返回值可以4种: 包含解析数据的字典 Item对象 新的Request对象(新的Requests也需要指定一个回调函数) 或者是可迭代对象(包含Items或Request) #3、在回调函数中解析页面内容 通常使用Scrapy自带的Selectors,但很明显你也可以使用Beutifulsoup,lxml或其他你爱用啥用啥。 #4、最后,针对返回的Items对象将会被持久化到数据库 通过Item Pipeline组件存到数据库:https://docs.scrapy.org/en/latest/topics/item-pipeline.html#topics-item-pipeline) 或者导出到不同的文件(通过Feed exports:https://docs.scrapy.org/en/latest/topics/feed-exports.html#topics-feed-exports)
Spiders总共提供了五种类
#1、scrapy.spiders.Spider #scrapy.Spider等同于scrapy.spiders.Spider #2、scrapy.spiders.CrawlSpider #3、scrapy.spiders.XMLFeedSpider #4、scrapy.spiders.CSVFeedSpider #5、scrapy.spiders.SitemapSpider
导入使用
import scrapy class AmazonSpider(scrapy.Spider): name = 'amazon' # 必须唯一 allowed_domains = ['www.amazon.cn'] # 允许域 start_urls = ['http://www.amazon.cn/'] # 如果你没有指定发送的请求地址,会默认使用只一个 def parse(self,response): pass
class scrapy.spiders.Spider
这是最简单的spider类,任何其他的spider类都需要继承它(包含你自己定义的)。
该类不提供任何特殊的功能,它仅提供了一个默认的start_requests方法默认从start_urls中读取url地址发送requests请求,并且默认parse作为回调函数.
import scrapy class AmazonSpider(scrapy.Spider): def __init__(self,keyword=None,*args,**kwargs): #在entrypoint文件里面传进来的keyword,在这里接收了 super(AmazonSpider,self).__init__(*args,**kwargs) self.keyword = keyword name = 'amazon' # 必须唯一 allowed_domains = ['www.amazon.cn'] # 允许域 start_urls = ['http://www.amazon.cn/'] # 如果你没有指定发送的请求地址,会默认使用只一个 custom_settings = { # 自定制配置文件,自己设置了用自己的,没有就找父类的 "BOT_NAME": 'HAIYAN_AMAZON', 'REQUSET_HEADERS': {}, } def start_requests(self): url = 'https://www.amazon.cn/s/ref=nb_sb_noss_1/461-4093573-7508641?' url+=urlencode({"field-keywords":self.keyword}) print(url) yield scrapy.Request( url, callback = self.parse_index, #指定回调函数 dont_filter = True, #不去重,这个也可以自己定制 # dont_filter = False, #去重,这个也可以自己定制 # meta={'a':1} #meta代理的时候会用 ) #如果要想测试自定义的dont_filter,可多返回结果重复的即可
DEFAULT_REQUEST_HEADERS = { # 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8', # 'Accept-Language': 'en', "User-Agent":"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/63.0.3239.132 Safari/537.36" }
1 #1、name = 'amazon' 2 定义爬虫名,scrapy会根据该值定位爬虫程序 3 所以它必须要有且必须唯一(In Python 2 this must be ASCII only.) 4 5 #2、allowed_domains = ['www.amazon.cn'] 6 定义允许爬取的域名,如果OffsiteMiddleware启动(默认就启动), 7 那么不属于该列表的域名及其子域名都不允许爬取 8 如果爬取的网址为:https://www.example.com/1.html,那就添加'example.com'到列表. 9 10 #3、start_urls = ['http://www.amazon.cn/'] 11 如果没有指定url,就从该列表中读取url来生成第一个请求 12 13 #4、custom_settings 14 值为一个字典,定义一些配置信息,在运行爬虫程序时,这些配置会覆盖项目级别的配置 15 所以custom_settings必须被定义成一个类属性,由于settings会在类实例化前被加载 16 17 #5、settings 18 通过self.settings['配置项的名字']可以访问settings.py中的配置,如果自己定义了custom_settings还是以自己的为准 19 20 #6、logger 21 日志名默认为spider的名字 22 self.logger.debug('=============>%s' %self.settings['BOT_NAME']) 23 24 #5、crawler:了解 25 该属性必须被定义到类方法from_crawler中 26 27 #6、from_crawler(crawler, *args, **kwargs):了解 28 You probably won’t need to override this directly because the default implementation acts as a proxy to the __init__() method, calling it with the given arguments args and named arguments kwargs. 29 30 #7、start_requests() 31 该方法用来发起第一个Requests请求,且必须返回一个可迭代的对象。它在爬虫程序打开时就被Scrapy调用,Scrapy只调用它一次。 32 默认从start_urls里取出每个url来生成Request(url, dont_filter=True) 33 34 #针对参数dont_filter,请看自定义去重规则 35 36 如果你想要改变起始爬取的Requests,你就需要覆盖这个方法,例如你想要起始发送一个POST请求,如下 37 class MySpider(scrapy.Spider): 38 name = 'myspider' 39 40 def start_requests(self): 41 return [scrapy.FormRequest("http://www.example.com/login", 42 formdata={'user': 'john', 'pass': 'secret'}, 43 callback=self.logged_in)] 44 45 def logged_in(self, response): 46 # here you would extract links to follow and return Requests for 47 # each of them, with another callback 48 pass 49 50 #8、parse(response) 51 这是默认的回调函数,所有的回调函数必须返回an iterable of Request and/or dicts or Item objects. 52 53 #9、log(message[, level, component]):了解 54 Wrapper that sends a log message through the Spider’s logger, kept for backwards compatibility. For more information see Logging from Spiders. 55 56 #10、closed(reason) 57 爬虫程序结束时自动触发
1 去重规则应该多个爬虫共享的,但凡一个爬虫爬取了,其他都不要爬了,实现方式如下 2 3 #方法一: 4 1、新增类属性 5 visited=set() #类属性 6 7 2、回调函数parse方法内: 8 def parse(self, response): 9 if response.url in self.visited: 10 return None 11 ....... 12 13 self.visited.add(response.url) 14 15 #方法一改进:针对url可能过长,所以我们存放url的hash值 16 def parse(self, response): 17 url=md5(response.request.url) 18 if url in self.visited: 19 return None 20 ....... 21 22 self.visited.add(url) 23 24 #方法二:Scrapy自带去重功能 25 配置文件: 26 DUPEFILTER_CLASS = 'scrapy.dupefilter.RFPDupeFilter' #默认的去重规则帮我们去重,去重规则在内存中 27 DUPEFILTER_DEBUG = False 28 JOBDIR = "保存范文记录的日志路径,如:/root/" # 最终路径为 /root/requests.seen,去重规则放文件中 29 30 scrapy自带去重规则默认为RFPDupeFilter,只需要我们指定 31 Request(...,dont_filter=False) ,如果dont_filter=True则告诉Scrapy这个URL不参与去重。 32 33 #方法三: 34 我们也可以仿照RFPDupeFilter自定义去重规则, 35 36 from scrapy.dupefilter import RFPDupeFilter,看源码,仿照BaseDupeFilter 37 38 #步骤一:在项目目录下自定义去重文件cumstomdupefilter.py 39 ''' 40 if hasattr("MyDupeFilter",from_settings): 41 func = getattr("MyDupeFilter",from_settings) 42 obj = func() 43 else: 44 return MyDupeFilter() 45 ''' 46 class MyDupeFilter(object): 47 def __init__(self): 48 self.visited = set() 49 50 @classmethod 51 def from_settings(cls, settings): 52 '''读取配置文件''' 53 return cls() 54 55 def request_seen(self, request): 56 '''请求看过没有,这个才是去重规则该调用的方法''' 57 if request.url in self.visited: 58 return True 59 self.visited.add(request.url) 60 61 def open(self): # can return deferred 62 '''打开的时候执行''' 63 pass 64 65 def close(self, reason): # can return a deferred 66 pass 67 68 def log(self, request, spider): # log that a request has been filtered 69 '''日志记录''' 70 pass 71 72 #步骤二:配置文件settings.py 73 # DUPEFILTER_CLASS = 'scrapy.dupefilter.RFPDupeFilter' #默认会去找这个类实现去重 74 #自定义去重规则 75 DUPEFILTER_CLASS = 'AMAZON.cumstomdupefilter.MyDupeFilter' 76 77 # 源码分析: 78 from scrapy.core.scheduler import Scheduler 79 见Scheduler下的enqueue_request方法:self.df.request_seen(request)
1 #例一: 2 import scrapy 3 4 class MySpider(scrapy.Spider): 5 name = 'example.com' 6 allowed_domains = ['example.com'] 7 start_urls = [ 8 'http://www.example.com/1.html', 9 'http://www.example.com/2.html', 10 'http://www.example.com/3.html', 11 ] 12 13 def parse(self, response): 14 self.logger.info('A response from %s just arrived!', response.url) 15 16 17 #例二:一个回调函数返回多个Requests和Items 18 import scrapy 19 20 class MySpider(scrapy.Spider): 21 name = 'example.com' 22 allowed_domains = ['example.com'] 23 start_urls = [ 24 'http://www.example.com/1.html', 25 'http://www.example.com/2.html', 26 'http://www.example.com/3.html', 27 ] 28 29 def parse(self, response): 30 for h3 in response.xpath('//h3').extract(): 31 yield {"title": h3} 32 33 for url in response.xpath('//a/@href').extract(): 34 yield scrapy.Request(url, callback=self.parse) 35 36 37 #例三:在start_requests()内直接指定起始爬取的urls,start_urls就没有用了, 38 39 import scrapy 40 from myproject.items import MyItem 41 42 class MySpider(scrapy.Spider): 43 name = 'example.com' 44 allowed_domains = ['example.com'] 45 46 def start_requests(self): 47 yield scrapy.Request('http://www.example.com/1.html', self.parse) 48 yield scrapy.Request('http://www.example.com/2.html', self.parse) 49 yield scrapy.Request('http://www.example.com/3.html', self.parse) 50 51 def parse(self, response): 52 for h3 in response.xpath('//h3').extract(): 53 yield MyItem(title=h3) 54 55 for url in response.xpath('//a/@href').extract(): 56 yield scrapy.Request(url, callback=self.parse) 57 58 59 #例四: 60 # -*- coding: utf-8 -*- 61 from urllib.parse import urlencode 62 # from scrapy.dupefilter import RFPDupeFilter 63 # from AMAZON.items import AmazonItem 64 from AMAZON.items import AmazonItem 65 ''' 66 67 spiders会循环做下面几件事 68 1、生成初始请求来爬取第一个urls,并且绑定一个回调函数 69 2、在回调函数中,解析response并且返回值 70 3、在回调函数中,解析页面内容(可通过Scrapy自带的Seletors或者BeautifuSoup等) 71 4、最后、针对返回的Items对象(就是你从返回结果中筛选出来自己想要的数据)将会被持久化到数据库 72 Spiders总共提供了五种类: 73 #1、scrapy.spiders.Spider #scrapy.Spider等同于scrapy.spiders.Spider 74 #2、scrapy.spiders.CrawlSpider 75 #3、scrapy.spiders.XMLFeedSpider 76 #4、scrapy.spiders.CSVFeedSpider 77 #5、scrapy.spiders.SitemapSpider 78 ''' 79 import scrapy 80 class AmazonSpider(scrapy.Spider): 81 def __init__(self,keyword=None,*args,**kwargs): #在entrypoint文件里面传进来的keyword,在这里接收了 82 super(AmazonSpider,self).__init__(*args,**kwargs) 83 self.keyword = keyword 84 85 name = 'amazon' # 必须唯一 86 allowed_domains = ['www.amazon.cn'] # 允许域 87 start_urls = ['http://www.amazon.cn/'] # 如果你没有指定发送的请求地址,会默认使用只一个 88 89 custom_settings = { # 自定制配置文件,自己设置了用自己的,没有就找父类的 90 "BOT_NAME": 'HAIYAN_AMAZON', 91 'REQUSET_HEADERS': {}, 92 } 93 94 def start_requests(self): 95 url = 'https://www.amazon.cn/s/ref=nb_sb_noss_1/461-4093573-7508641?' 96 url+=urlencode({"field-keywords":self.keyword}) 97 print(url) 98 yield scrapy.Request( 99 url, 100 callback = self.parse_index, #指定回调函数 101 dont_filter = True, #不去重,这个也可以自己定制 102 # dont_filter = False, #去重,这个也可以自己定制 103 # meta={'a':1} #meta代理的时候会用 104 ) 105 #如果要想测试自定义的dont_filter,可多返回结果重复的即可 106 107 108 109 def parse_index(self, response): 110 '''获取详情页和下一页的链接''' 111 detail_urls = response.xpath('//*[contains(@id,"result_")]/div/div[3]/div[1]/a/@href').extract() 112 print(detail_urls) 113 # print("%s 解析 %s",(response.url,(len(response.body)))) 114 for detail_url in detail_urls: 115 yield scrapy.Request( 116 url=detail_url, 117 callback=self.parse_detail #记得每次返回response的时候记得绑定一个回调函数 118 ) 119 next_url = response.urljoin(response.xpath(response.xpath('//*[@id="pagnNextLink"]/@href').extract_first())) 120 # 因为下一页的url是不完整的,用urljoin就可以吧路径前缀拿到并且拼接 121 # print(next_url) 122 yield scrapy.Request( 123 url=next_url, 124 callback=self.parse_index #因为下一页也属于是索引页,让去解析索引页 125 ) 126 def parse_detail(self,response): 127 '''详情页解析''' 128 name = response.xpath('//*[@id="productTitle"]/text()').extract_first().strip()#获取name 129 price = response.xpath('//*[@id="price"]//*[@class="a-size-medium a-color-price"]/text()').extract_first()#获取价格 130 delivery_method=''.join(response.xpath('//*[@id="ddmMerchantMessage"]//text()').extract()) #获取配送方式 131 print(name) 132 print(price) 133 print(delivery_method) 134 #上面是筛选出自己想要的项 135 #必须返回一个Item对象,那么这个item对象,是从item.py中来,和django中的model类似, 136 # 但是这里的item对象也可当做是一个字典,和字典的操作一样 137 item = AmazonItem()# 实例化 138 item["name"] = name 139 item["price"] = price 140 item["delivery_method"] = delivery_method 141 return item 142 143 def close(spider, reason): 144 print("结束啦")
1 我们可能需要在命令行为爬虫程序传递参数,比如传递初始的url,像这样 2 #命令行执行 3 scrapy crawl myspider -a category=electronics 4 5 #在__init__方法中可以接收外部传进来的参数 6 import scrapy 7 8 class MySpider(scrapy.Spider): 9 name = 'myspider' 10 11 def __init__(self, category=None, *args, **kwargs): 12 super(MySpider, self).__init__(*args, **kwargs) 13 self.start_urls = ['http://www.example.com/categories/%s' % category] 14 #... 15 16 17 #注意接收的参数全都是字符串,如果想要结构化的数据,你需要用类似json.loads的方法
Selectors选择器
response.selector.css() response.selector.xpath() 可简写为 response.css() response.xpath() #1 //与/ response.xpath('//body/a/')# response.css('div a::text') >>> response.xpath('//body/a') #开头的//代表从整篇文档中寻找,body之后的/代表body的儿子 [] >>> response.xpath('//body//a') #开头的//代表从整篇文档中寻找,body之后的//代表body的子子孙孙 [<Selector xpath='//body//a' data='<a href="image1.html">Name: My image 1 <'>, <Selector xpath='//body//a' data='<a href="image2.html">Name: My image 2 <'>, <Selector xpath='//body//a' data='<a href=" image3.html">Name: My image 3 <'>, <Selector xpath='//body//a' data='<a href="image4.html">Name: My image 4 <'>, <Selector xpath='//body//a' data='<a href="image5.html">Name: My image 5 <'>] #2 text >>> response.xpath('//body//a/text()') >>> response.css('body a::text') #3、extract与extract_first:从selector对象中解出内容 >>> response.xpath('//div/a/text()').extract() ['Name: My image 1 ', 'Name: My image 2 ', 'Name: My image 3 ', 'Name: My image 4 ', 'Name: My image 5 '] >>> response.css('div a::text').extract() ['Name: My image 1 ', 'Name: My image 2 ', 'Name: My image 3 ', 'Name: My image 4 ', 'Name: My image 5 '] >>> response.xpath('//div/a/text()').extract_first() 'Name: My image 1 ' >>> response.css('div a::text').extract_first() 'Name: My image 1 ' #4、属性:xpath的属性加前缀@ >>> response.xpath('//div/a/@href').extract_first() 'image1.html' >>> response.css('div a::attr(href)').extract_first() 'image1.html' #4、嵌套查找 >>> response.xpath('//div').css('a').xpath('@href').extract_first() 'image1.html' #5、设置默认值 >>> response.xpath('//div[@id="xxx"]').extract_first(default="not found") 'not found' #4、按照属性查找 response.xpath('//div[@id="images"]/a[@href="image3.html"]/text()').extract() response.css('#images a[@href="image3.html"]/text()').extract() #5、按照属性模糊查找 response.xpath('//a[contains(@href,"image")]/@href').extract() response.css('a[href*="image"]::attr(href)').extract() response.xpath('//a[contains(@href,"image")]/img/@src').extract() response.css('a[href*="imag"] img::attr(src)').extract() response.xpath('//*[@href="image1.html"]') response.css('*[href="image1.html"]') #6、正则表达式 response.xpath('//a/text()').re(r'Name: (.*)') response.xpath('//a/text()').re_first(r'Name: (.*)') #7、xpath相对路径 >>> res=response.xpath('//a[contains(@href,"3")]')[0] >>> res.xpath('img') [<Selector xpath='img' data='<img src="image3_thumb.jpg">'>] >>> res.xpath('./img') [<Selector xpath='./img' data='<img src="image3_thumb.jpg">'>] >>> res.xpath('.//img') [<Selector xpath='.//img' data='<img src="image3_thumb.jpg">'>] >>> res.xpath('//img') #这就是从头开始扫描 [<Selector xpath='//img' data='<img src="image1_thumb.jpg">'>, <Selector xpath='//img' data='<img src="image2_thumb.jpg">'>, <Selector xpath='//img' data='<img src="image3_thumb.jpg">'>, <Selector xpa th='//img' data='<img src="image4_thumb.jpg">'>, <Selector xpath='//img' data='<img src="image5_thumb.jpg">'>] #8、带变量的xpath >>> response.xpath('//div[@id=$xxx]/a/text()',xxx='images').extract_first() 'Name: My image 1 ' >>> response.xpath('//div[count(a)=$yyy]/@id',yyy=5).extract_first() #求有5个a标签的div的id 'images'
Item Pipeline
自定义pipeline
#一:可以写多个Pipeline类 #1、如果优先级高的Pipeline的process_item返回一个值或者None,会自动传给下一个pipline的process_item, #2、如果只想让第一个Pipeline执行,那得让第一个pipline的process_item抛出异常raise DropItem() #3、可以用spider.name == '爬虫名' 来控制哪些爬虫用哪些pipeline 二:示范 from scrapy.exceptions import DropItem class CustomPipeline(object): def __init__(self,v): self.value = v @classmethod def from_crawler(cls, crawler): """ Scrapy会先通过getattr判断我们是否自定义了from_crawler,有则调它来完 成实例化 """ val = crawler.settings.getint('MMMM') return cls(val) def open_spider(self,spider): """ 爬虫刚启动时执行一次 """ print('000000') def close_spider(self,spider): """ 爬虫关闭时执行一次 """ print('111111') def process_item(self, item, spider): # 操作并进行持久化 # return表示会被后续的pipeline继续处理 return item # 表示将item丢弃,不会被后续pipeline处理 # raise DropItem()
1 ''' 2 #1、settings.py 3 HOST="127.0.0.1" 4 PORT=27017 5 USER="root" 6 PWD="123" 7 DB="amazon" 8 TABLE="goods" 9 10 ''' 11 from scrapy.exceptions import DropItem 12 from pymongo import MongoClient 13 14 class MongoPipeline(object): 15 '''2、把解析好的item对象做一个持久化,保存到数据库中''' 16 def __init__(self,db,collection,host,port,user,pwd): 17 self.db = db 18 self.collection = collection #文档(表) 19 self.host = host 20 self.port = port 21 self.user = user 22 self.pwd = pwd 23 24 @classmethod 25 def from_crawler(cls,crawler): 26 '''1、Scrapy会先通过getattr判断我们是否自定义了from_crawler,有则调它来完 27 成实例化''' 28 db = crawler.settings.get("DB") 29 collection = crawler.settings.get("COLLECTION") 30 host = crawler.settings.get("HOST") 31 port = crawler.settings.get("PORT") 32 user = crawler.settings.get("USER") 33 pwd = crawler.settings.get("PWD") 34 return cls(db,collection,host,port,user,pwd) #cls是当前的类,类加括号执行__init__方法 35 36 def open_spider(self,spider): 37 '''3、爬虫刚启动时执行一次''' 38 print('==============>爬虫程序刚刚启动') 39 self.client = MongoClient('mongodb://%s:%s@%s:%s'%( 40 self.user, 41 self.pwd, 42 self.host, 43 self.port 44 )) 45 46 def close_spider(self,spider): 47 '''5、关闭爬虫程序''' 48 print('==============>爬虫程序运行完毕') 49 self.client.close() 50 51 def process_item(self, item, spider): 52 '''4、操作并执行持久化''' 53 # return表示会被后续的pipeline继续处理 54 d = dict(item) 55 if all(d.values()): 56 self.client[self.db][self.collection].save(d) #保存到数据库 57 return item 58 # 表示将item丢弃,不会被后续pipeline处理 59 # raise DropItem() 60 61 62 63 class FilePipeline(object): 64 def __init__(self, file_path): 65 self.file_path=file_path 66 67 @classmethod 68 def from_crawler(cls, crawler): 69 """ 70 Scrapy会先通过getattr判断我们是否自定义了from_crawler,有则调它来完 71 成实例化 72 """ 73 file_path = crawler.settings.get('FILE_PATH') 74 75 76 return cls(file_path) 77 78 def open_spider(self, spider): 79 """ 80 爬虫刚启动时执行一次 81 """ 82 print('==============>爬虫程序刚刚启动') 83 self.fileobj=open(self.file_path,'w',encoding='utf-8') 84 85 def close_spider(self, spider): 86 """ 87 爬虫关闭时执行一次 88 """ 89 print('==============>爬虫程序运行完毕') 90 self.fileobj.close() 91 92 def process_item(self, item, spider): 93 # 操作并进行持久化 94 95 # return表示会被后续的pipeline继续处理 96 d = dict(item) 97 if all(d.values()): 98 self.fileobj.write(r"%s\n" %str(d)) 99 100 return item 101 102 # 表示将item丢弃,不会被后续pipeline处理 103 # raise DropItem()
Downloader Middleware下载中间件
用途
1、在process——request内,自定义下载,不用scrapy的下载 2、对请求进行二次加工,比如 设置请求头 设置cookie 添加代理 scrapy自带的代理组件: from scrapy.downloadermiddlewares.httpproxy import HttpProxyMiddleware from urllib.request import getproxies
写下载器中间件
1 class DownMiddleware1(object): 2 def process_request(self, request, spider): 3 """ 4 请求需要被下载时,经过所有下载器中间件的process_request调用 5 :param request: 6 :param spider: 7 :return: 8 None,继续后续中间件去下载; 9 Response对象,停止process_request的执行,开始执行process_response 10 Request对象,停止中间件的执行,将Request重新调度器 11 raise IgnoreRequest异常,停止process_request的执行,开始执行process_exception 12 """ 13 pass 14 15 16 17 def process_response(self, request, response, spider): 18 """ 19 spider处理完成,返回时调用 20 :param response: 21 :param result: 22 :param spider: 23 :return: 24 Response 对象:转交给其他中间件process_response 25 Request 对象:停止中间件,request会被重新调度下载 26 raise IgnoreRequest 异常:调用Request.errback 27 """ 28 print('response1') 29 return response 30 31 def process_exception(self, request, exception, spider): 32 """ 33 当下载处理器(download handler)或 process_request() (下载中间件)抛出异常 34 :param response: 35 :param exception: 36 :param spider: 37 :return: 38 None:继续交给后续中间件处理异常; 39 Response对象:停止后续process_exception方法 40 Request对象:停止中间件,request将会被重新调用下载 41 """ 42 return None
配置
DOWNLOADER_MIDDLEWARES = { #'xdb.middlewares.XdbDownloaderMiddleware': 543, # 'xdb.proxy.XdbProxyMiddleware':751, 'xdb.md.Md1':666, 'xdb.md.Md2':667, }
应用
- user-agent - 代理
Spider Middleware爬虫中间件
1 class SpiderMiddleware(object): 2 3 def process_spider_input(self,response, spider): 4 """ 5 下载完成,执行,然后交给parse处理 6 :param response: 7 :param spider: 8 :return: 9 """ 10 pass 11 12 def process_spider_output(self,response, result, spider): 13 """ 14 spider处理完成,返回时调用 15 :param response: 16 :param result: 17 :param spider: 18 :return: 必须返回包含 Request 或 Item 对象的可迭代对象(iterable) 19 """ 20 return result 21 22 def process_spider_exception(self,response, exception, spider): 23 """ 24 异常调用 25 :param response: 26 :param exception: 27 :param spider: 28 :return: None,继续交给后续中间件处理异常;含 Response 或 Item 的可迭代对象(iterable),交给调度器或pipeline 29 """ 30 return None 31 32 33 def process_start_requests(self,start_requests, spider): 34 """ 35 爬虫启动时调用 36 :param start_requests: 37 :param spider: 38 :return: 包含 Request 对象的可迭代对象 39 """ 40 return start_requests
配置
SPIDER_MIDDLEWARES = { # 'xdb.middlewares.XdbSpiderMiddleware': 543, 'xdb.sd.Sd1': 666, 'xdb.sd.Sd2': 667, }
应用
- 深度 - 优先级
自定制命令
单爬虫运行
import sys from scrapy.cmdline import execute if __name__ == '__main__': execute(["scrapy","crawl","chouti","--nolog"])
所有爬虫
- 在spiders同级创建任意目录,如:commands
- 在其中创建 crawlall.py 文件 (此处文件名就是自定义的命令)
from scrapy.commands import ScrapyCommand from scrapy.utils.project import get_project_settings class Command(ScrapyCommand): requires_project = True def syntax(self): return '[options]' def short_desc(self): return 'Runs all of the spiders' def run(self, args, opts): spider_list = self.crawler_process.spiders.list() for name in spider_list: self.crawler_process.crawl(name, **opts.__dict__) self.crawler_process.start()
- 在settings.py 中添加配置 COMMANDS_MODULE = '项目名称.目录名称'
- 在项目目录执行命令:scrapy crawlall
自定义扩展(信号)
scrapy自定义扩展的好处是可以在任意我们想要的位置添加功能,而其他组件中提供的功能只能在规定的位置执行
1 #1、在与settings同级目录下新建一个文件,文件名可以为extentions.py,内容如下 2 from scrapy import signals 3 4 5 class MyExtension(object): 6 def __init__(self, value): 7 self.value = value 8 9 @classmethod 10 def from_crawler(cls, crawler): 11 val = crawler.settings.getint('MMMM') 12 obj = cls(val) 13 14 crawler.signals.connect(obj.spider_opened, signal=signals.spider_opened) 15 crawler.signals.connect(obj.spider_closed, signal=signals.spider_closed) 16 17 return obj 18 19 def spider_opened(self, spider): 20 print('=============>open') 21 22 def spider_closed(self, spider): 23 print('=============>close') 24 25 #2、配置生效 26 EXTENSIONS = { 27 "Amazon.extentions.MyExtension":200 28 }
settings.py
1 #==>第一部分:基本配置<=== 2 #1、项目名称,默认的USER_AGENT由它来构成,也作为日志记录的日志名 3 BOT_NAME = 'Amazon' 4 5 #2、爬虫应用路径 6 SPIDER_MODULES = ['Amazon.spiders'] 7 NEWSPIDER_MODULE = 'Amazon.spiders' 8 9 #3、客户端User-Agent请求头 10 #USER_AGENT = 'Amazon (+http://www.yourdomain.com)' 11 12 #4、是否遵循爬虫协议 13 # Obey robots.txt rules 14 ROBOTSTXT_OBEY = False 15 16 #5、是否支持cookie,cookiejar进行操作cookie,默认开启 17 #COOKIES_ENABLED = False 18 19 #6、Telnet用于查看当前爬虫的信息,操作爬虫等...使用telnet ip port ,然后通过命令操作 20 #TELNETCONSOLE_ENABLED = False 21 #TELNETCONSOLE_HOST = '127.0.0.1' 22 #TELNETCONSOLE_PORT = [6023,] 23 24 #7、Scrapy发送HTTP请求默认使用的请求头 25 #DEFAULT_REQUEST_HEADERS = { 26 # 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8', 27 # 'Accept-Language': 'en', 28 #} 29 30 31 32 #===>第二部分:并发与延迟<=== 33 #1、下载器总共最大处理的并发请求数,默认值16 34 #CONCURRENT_REQUESTS = 32 35 36 #2、每个域名能够被执行的最大并发请求数目,默认值8 37 #CONCURRENT_REQUESTS_PER_DOMAIN = 16 38 39 #3、能够被单个IP处理的并发请求数,默认值0,代表无限制,需要注意两点 40 #I、如果不为零,那CONCURRENT_REQUESTS_PER_DOMAIN将被忽略,即并发数的限制是按照每个IP来计算,而不是每个域名 41 #II、该设置也影响DOWNLOAD_DELAY,如果该值不为零,那么DOWNLOAD_DELAY下载延迟是限制每个IP而不是每个域 42 #CONCURRENT_REQUESTS_PER_IP = 16 43 44 #4、如果没有开启智能限速,这个值就代表一个规定死的值,代表对同一网址延迟请求的秒数 45 #DOWNLOAD_DELAY = 3 46 47 48 49 #===>第三部分:智能限速/自动节流:AutoThrottle extension<=== 50 #一:介绍 51 from scrapy.contrib.throttle import AutoThrottle #http://scrapy.readthedocs.io/en/latest/topics/autothrottle.html#topics-autothrottle 52 设置目标: 53 1、比使用默认的下载延迟对站点更好 54 2、自动调整scrapy到最佳的爬取速度,所以用户无需自己调整下载延迟到最佳状态。用户只需要定义允许最大并发的请求,剩下的事情由该扩展组件自动完成 55 56 57 #二:如何实现? 58 在Scrapy中,下载延迟是通过计算建立TCP连接到接收到HTTP包头(header)之间的时间来测量的。 59 注意,由于Scrapy可能在忙着处理spider的回调函数或者无法下载,因此在合作的多任务环境下准确测量这些延迟是十分苦难的。 不过,这些延迟仍然是对Scrapy(甚至是服务器)繁忙程度的合理测量,而这扩展就是以此为前提进行编写的。 60 61 62 #三:限速算法 63 自动限速算法基于以下规则调整下载延迟 64 #1、spiders开始时的下载延迟是基于AUTOTHROTTLE_START_DELAY的值 65 #2、当收到一个response,对目标站点的下载延迟=收到响应的延迟时间/AUTOTHROTTLE_TARGET_CONCURRENCY 66 #3、下一次请求的下载延迟就被设置成:对目标站点下载延迟时间和过去的下载延迟时间的平均值 67 #4、没有达到200个response则不允许降低延迟 68 #5、下载延迟不能变的比DOWNLOAD_DELAY更低或者比AUTOTHROTTLE_MAX_DELAY更高 69 70 #四:配置使用 71 #开启True,默认False 72 AUTOTHROTTLE_ENABLED = True 73 #起始的延迟 74 AUTOTHROTTLE_START_DELAY = 5 75 #最小延迟 76 DOWNLOAD_DELAY = 3 77 #最大延迟 78 AUTOTHROTTLE_MAX_DELAY = 10 79 #每秒并发请求数的平均值,不能高于 CONCURRENT_REQUESTS_PER_DOMAIN或CONCURRENT_REQUESTS_PER_IP,调高了则吞吐量增大强奸目标站点,调低了则对目标站点更加”礼貌“ 80 #每个特定的时间点,scrapy并发请求的数目都可能高于或低于该值,这是爬虫视图达到的建议值而不是硬限制 81 AUTOTHROTTLE_TARGET_CONCURRENCY = 16.0 82 #调试 83 AUTOTHROTTLE_DEBUG = True 84 CONCURRENT_REQUESTS_PER_DOMAIN = 16 85 CONCURRENT_REQUESTS_PER_IP = 16 86 87 88 89 #===>第四部分:爬取深度与爬取方式<=== 90 #1、爬虫允许的最大深度,可以通过meta查看当前深度;0表示无深度 91 # DEPTH_LIMIT = 3 92 93 #2、爬取时,0表示深度优先Lifo(默认);1表示广度优先FiFo 94 95 # 后进先出,深度优先 96 # DEPTH_PRIORITY = 0 97 # SCHEDULER_DISK_QUEUE = 'scrapy.squeue.PickleLifoDiskQueue' 98 # SCHEDULER_MEMORY_QUEUE = 'scrapy.squeue.LifoMemoryQueue' 99 # 先进先出,广度优先 100 101 # DEPTH_PRIORITY = 1 102 # SCHEDULER_DISK_QUEUE = 'scrapy.squeue.PickleFifoDiskQueue' 103 # SCHEDULER_MEMORY_QUEUE = 'scrapy.squeue.FifoMemoryQueue' 104 105 106 #3、调度器队列 107 # SCHEDULER = 'scrapy.core.scheduler.Scheduler' 108 # from scrapy.core.scheduler import Scheduler 109 110 #4、访问URL去重 111 # DUPEFILTER_CLASS = 'step8_king.duplication.RepeatUrl' 112 113 114 115 #===>第五部分:中间件、Pipelines、扩展<=== 116 #1、Enable or disable spider middlewares 117 # See http://scrapy.readthedocs.org/en/latest/topics/spider-middleware.html 118 #SPIDER_MIDDLEWARES = { 119 # 'Amazon.middlewares.AmazonSpiderMiddleware': 543, 120 #} 121 122 #2、Enable or disable downloader middlewares 123 # See http://scrapy.readthedocs.org/en/latest/topics/downloader-middleware.html 124 DOWNLOADER_MIDDLEWARES = { 125 # 'Amazon.middlewares.DownMiddleware1': 543, 126 } 127 128 #3、Enable or disable extensions 129 # See http://scrapy.readthedocs.org/en/latest/topics/extensions.html 130 #EXTENSIONS = { 131 # 'scrapy.extensions.telnet.TelnetConsole': None, 132 #} 133 134 #4、Configure item pipelines 135 # See http://scrapy.readthedocs.org/en/latest/topics/item-pipeline.html 136 ITEM_PIPELINES = { 137 # 'Amazon.pipelines.CustomPipeline': 200, 138 } 139 140 141 142 #===>第六部分:缓存<=== 143 """ 144 1. 启用缓存 145 目的用于将已经发送的请求或相应缓存下来,以便以后使用 146 147 from scrapy.downloadermiddlewares.httpcache import HttpCacheMiddleware 148 from scrapy.extensions.httpcache import DummyPolicy 149 from scrapy.extensions.httpcache import FilesystemCacheStorage 150 """ 151 # 是否启用缓存策略 152 # HTTPCACHE_ENABLED = True 153 154 # 缓存策略:所有请求均缓存,下次在请求直接访问原来的缓存即可 155 # HTTPCACHE_POLICY = "scrapy.extensions.httpcache.DummyPolicy" 156 # 缓存策略:根据Http响应头:Cache-Control、Last-Modified 等进行缓存的策略 157 # HTTPCACHE_POLICY = "scrapy.extensions.httpcache.RFC2616Policy" 158 159 # 缓存超时时间 160 # HTTPCACHE_EXPIRATION_SECS = 0 161 162 # 缓存保存路径 163 # HTTPCACHE_DIR = 'httpcache' 164 165 # 缓存忽略的Http状态码 166 # HTTPCACHE_IGNORE_HTTP_CODES = [] 167 168 # 缓存存储的插件 169 # HTTPCACHE_STORAGE = 'scrapy.extensions.httpcache.FilesystemCacheStorage'