五、Scrapy框架(2)——快速入门

Scrapy快速入门

安装和文档:

  1. 安装:通过pip install scrapy即可安装。
  2. Scrapy官方文档:http://doc.scrapy.org/en/latest
  3. Scrapy中文文档:http://scrapy-chs.readthedocs.io/zh_CN/latest/index.html

注意:

  1. ubuntu上安装scrapy之前,需要先安装以下依赖:
    sudo apt-get install python3-dev build-essential python3-pip libxml2-dev libxslt1-dev zlib1g-dev libffi-dev libssl-dev,然后再通过pip install scrapy安装。
  2. 如果在windows系统下,提示这个错误ModuleNotFoundError: No module named 'win32api',那么使用以下命令可以解决:pip install pypiwin32

快速入门:

创建项目:

要使用Scrapy框架创建项目,需要通过命令来创建。首先进入到你想把这个项目存放的目录。然后使用以下命令创建:

scrapy startproject [项目名称]

目录结构介绍:

以下介绍下主要文件的作用:

  1. items.py:用来存放爬虫爬取下来数据的模型。
  2. middlewares.py:用来存放各种中间件的文件。
  3. pipelines.py:用来将items的模型存储到本地磁盘中。
  4. settings.py:本爬虫的一些配置信息(比如请求头、多久发送一次请求、ip代理池等)。
  5. scrapy.cfg:项目的配置文件。
  6. spiders包:以后所有的爬虫,都是存放到这个里面。

使用Scrapy框架爬取糗事百科段子:

使用命令创建一个爬虫:

scrapy gensipder qsbk "qiushibaike.com"

创建了一个名字叫做qsbk的爬虫,并且能爬取的网页只会限制在qiushibaike.com这个域名下。

爬虫代码解析:

import scrapy

class QsbkSpider(scrapy.Spider):
    name = 'qsbk'
    allowed_domains = ['qiushibaike.com']
    start_urls = ['http://www.qiushibaike.com/']

    def parse(self, response):
        pass

其实这些代码我们完全可以自己手动去写,而不用命令。只不过是不用命令,自己写这些代码比较麻烦。
要创建一个Spider,那么必须自定义一个类,继承自scrapy.Spider,然后在这个类中定义三个属性和一个方法。

  1. name:这个爬虫的名字,名字必须是唯一的。
  2. allow_domains:允许的域名。爬虫只会爬取这个域名下的网页,其他不是这个域名下的网页会被自动忽略。
  3. start_urls:爬虫从这个变量中的url开始。
  4. parse:引擎会把下载器下载回来的数据扔给爬虫解析,爬虫再把数据传给这个parse方法。这个是个固定的写法。这个方法的作用有两个,第一个是提取想要的数据。第二个是生成下一个请求的url。

修改settings.py代码:

在做一个爬虫之前,一定要记得修改setttings.py中的设置。两个地方是强烈建议设置的。

  1. ROBOTSTXT_OBEY设置为False。默认是True。即遵守机器协议,那么在爬虫的时候,scrapy首先去找robots.txt文件,如果没有找到。则直接停止爬取。
  2. DEFAULT_REQUEST_HEADERS添加User-Agent。这个也是告诉服务器,我这个请求是一个正常的请求,不是一个爬虫。

完成的爬虫代码:

  1. 爬虫部分代码:

    # -*- coding: utf-8 -*-
    import scrapy
    from qsbk.qsbk.items import QsbkItem
    from scrapy.http.response.text import Response
    
    
    class QsbkSpiderSpider(scrapy.Spider):
        name = 'qsbk_spider'
        allowed_domains = ['qiushibaike.com']
        start_urls = ['https://www.qiushibaike.com/text/']
    
        def parse(self, response):
            # selectorList
            duanzidivs = response.xpath("//div[@id='content-left']/div")
            for duanzidiv in duanzidivs:
                # selector
                author = duanzidiv.xpath(".//h2/text()").get().strip()
                content = duanzidiv.xpath(".//div[@class='content']//text()").getall()
                content = "".join(content).strip()
                item = QsbkItem(author=author,content=content)
                yield item
    
    
  2. items.py部分代码:

     import scrapy
     class QsbkItem(scrapy.Item):
         author = scrapy.Field()
         content = scrapy.Field()
    
  3. pipeline部分代码:

    import json
    
    
    class QsbkPipeline(object):
        def __init__(self):
            self.fp = open("duanzi.json", 'w', encoding='utf-8')
    
        def open_spider(self, spider):
            print('爬虫开始了...')
    
        def process_item(self, item, spider):
            item_json = json.dumps(dict(item), ensure_ascii=False)
            self.fp.write(item_json + '\n')
            return item
    
        def close_spider(self, spider):
            self.fp.close()
            print('爬虫结束了...')
    

运行scrapy项目:

运行scrapy项目。需要在终端,进入项目所在的路径,然后scrapy crawl [爬虫名字]即可运行指定的爬虫。如果不想每次都在命令行中运行,那么可以把这个命令写在一个文件中。以后就在pycharm中执行运行这个文件就可以了。比如现在新创建一个文件叫做start.py,然后在这个文件中填入以下代码:

from scrapy import cmdline

cmdline.execute("scrapy crawl qsbk".split())

糗事百科Scrapy爬虫笔记:

  1. response是一个scrapy.http.response.html.HtmlResponse对象。可以执行xpathcss语法来提取数据。
  2. 提取出来的数据,是一个Selector或者是一个SelectorList对象。如果想要获取其中的字符串。
    那么应该执行getall或者get方法。
  3. getall方法:获取Selector中的所有文本。返回的是一个列表
  4. get方法:获取的是Selector中的第一个文本,返回的是一个str类型。
  5. 如果数据解析回来,要传给pipeline处理,那么可以使用 yield来返回。或者是收集所有的item。最后统一使用return返回。
  6. item:建议在items.py中定义好模型,以后就不要使用字典。
  7. pipeline:这个是专门用来保存数据的。其中有三个方法是会经常用的。
  • open_spider(self,spider):当爬虫被打开的时候执行。
  • process_items(self,item,spider):当爬虫有item传过来的时候会被调用。
  • close_spider(self,spider):当爬虫关闭的时候会被调用

要激活pipeline,应该在settings.py中,设置ITEM_PIPELINES。示例如下:

ITEM_PIPELINES = {
   'qsbk.pipelines.QsbkPipeline': 300,
}

JsonItemExporter 和 JsonLinesItemExporter

保存 json 数据的时候,可以使用这两个类,让操作变得更简单。

  1. JsonItemExporter:这个是每次把数据添加到内存中,最后统一写入到磁盘中。好处是:存储的数据是一个满足 json 规则的数据,坏处是如果数据量比较大,那么比较耗内存。示例代码如下:

    from scrapy.exporters import JsonItemExporter
    
    
    class QsbkPipeline(object):
        def __init__(self):
            self.fp = open("duanzi.json", 'wb')
            self.exporter = JsonItemExporter(self.fp,ensure_ascii=False,encoding='utf-8')
            self.exporter.start_exporting()
    
        def open_spider(self, spider):
            print('爬虫开始了...')
    
        def process_item(self, item, spider):
            self.exporter.export_item(item)
            return item
    
        def close_spider(self, spider):
            self.exporter.finish_exporting()
            self.fp.close()
            print('爬虫结束了...')
    
  2. JsonLinesItemExporter:这个是每次调用export_item`的时候就把这个item存储到硬盘中。好处是每次处理数据的时候就直接存储到了硬盘中,这样不会耗内存,数据也比较安全。坏处是每一个字典是一行。整个文件不是一个满足json格式的文件。示例代码如下:

    from scrapy.exporters import JsonLinesItemExporter
    
    class QsbkPipeline(object):
        def __init__(self):
            self.fp = open("duanzi3.json", 'wb')
            self.exporter = JsonLinesItemExporter(self.fp,ensure_ascii=False,encoding='utf-8')
    
        def open_spider(self, spider):
            print('爬虫开始了...')
    
        def process_item(self, item, spider):
            self.exporter.export_item(item)
            return item
    
        def close_spider(self, spider):
            self.fp.close()
            print('爬虫结束了...')
    
posted @ 2019-04-30 21:02  渣爷  阅读(202)  评论(0)    收藏  举报