1.基于终端指令的持久化存储

保证爬虫文件的parse方法中有可迭代类型对象(通常为列表or字典)的返回,该返回值可以通过终端指令的形式写入指定格式的文件中进行持久化操作。

  1. 执行输出指定格式进行存储:将爬取到的数据写入不同格式的文件中进行存储
  2.  
  3. scrapy crawl 爬虫名称 -o xxx.json
  4.  
  5. scrapy crawl 爬虫名称 -o xxx.xml
  6.  
  7. scrapy crawl 爬虫名称 -o xxx.csv

2.基于管道的持久化存储

scrapy框架中已经为我们专门集成好了高效、便捷的持久化操作功能,我们直接使用即可。要想使用scrapy的持久化操作功能,我们首先来认识如下两个文件:

  1. items.py:数据结构模板文件。定义数据属性。
  2.  
  3. pipelines.py:管道文件。接收数据(items),进行持久化操作。
  4.  
  5. 持久化流程:
  6.  
  7. 1.爬虫文件爬取到数据后,需要将数据封装到items对象中。
  8.  
  9. 2.使用yield关键字将items对象提交给pipelines管道进行持久化操作。
  10.  
  11. 3.在管道文件中的process_item方法中接收爬虫文件提交过来的item对象,然后编写持久化存储的代码将item对象中存储的数据进行持久化存储
  12.  
  13. 4.settings.py配置文件中开启管道

小试牛刀:将糗事百科首页中的段子和作者数据爬取下来,然后进行持久化存储

- 爬虫文件:qiubaiDemo.py

  1. # -*- coding: utf-8 -*-
  2.  
  3. import scrapy
  4.  
  5. from secondblood.items import SecondbloodItem
  6.  
  7. class QiubaidemoSpider(scrapy.Spider):
  8.  
  9. name = 'qiubaiDemo'
  10.  
  11. allowed_domains = ['www.qiushibaike.com']
  12.  
  13. start_urls = ['http://www.qiushibaike.com/']
  14.  
  15. def parse(self, response):
  16.  
  17. odiv = response.xpath('//div[@id="content-left"]/div')
  18.  
  19. for div in odiv:
  20.  
  21. # xpath函数返回的为列表,列表中存放的数据为Selector类型的数据。我们解析到的内容被封装在了Selector对象中,需要调用extract()函数将解析的内容从Selecor中取出。
  22.  
  23. author = div.xpath('.//div[@class="author clearfix"]//h2/text()').extract_first()
  24.  
  25. author = author.strip('\n')#过滤空行
  26.  
  27. content = div.xpath('.//div[@class="content"]/span/text()').extract_first()
  28.  
  29. content = content.strip('\n')#过滤空行
  30.  
  31. #将解析到的数据封装至items对象中
  32.  
  33. item = SecondbloodItem()
  34.  
  35. item['author'] = author
  36.  
  37. item['content'] = content
  38.  
  39. yield item#提交item到管道文件(pipelines.py)

- items文件:items.py

  1. import scrapy
  2.  
  3. class SecondbloodItem(scrapy.Item):
  4.  
  5. # define the fields for your item here like:
  6.  
  7. # name = scrapy.Field()
  8.  
  9. author = scrapy.Field() #存储作者
  10.  
  11. content = scrapy.Field() #存储段子内容

- 管道文件:pipelines.py

  1. # -*- coding: utf-8 -*-
  2.  
  3. # Define your item pipelines here
  4.  
  5. # Don't forget to add your pipeline to the ITEM_PIPELINES setting
  6.  
  7. # See: https://doc.scrapy.org/en/latest/topics/item-pipeline.html
  8.  
  9. class SecondbloodPipeline(object):
  10.  
  11. #构造方法
  12.  
  13. def __init__(self):
  14.  
  15. self.fp = None #定义一个文件描述符属性
  16.  
  17.   #下列都是在重写父类的方法:
  18.  
  19. #开始爬虫时,执行一次
  20.  
  21. def open_spider(self,spider):
  22.  
  23. print('爬虫开始')
  24.  
  25. self.fp = open('./data.txt', 'w')
  26.  
  27.    #因为该方法会被执行调用多次,所以文件的开启和关闭操作写在了另外两个只会各自执行一次的方法中。
  28.  
  29. def process_item(self, item, spider):
  30.  
  31. #将爬虫程序提交的item进行持久化存储
  32.  
  33. self.fp.write(item['author'] + ':' + item['content'] + '\n')
  34.  
  35. return item
  36.  
  37. #结束爬虫时,执行一次
  38.  
  39. def close_spider(self,spider):
  40.  
  41. self.fp.close()
  42.  
  43. print('爬虫结束')

- 配置文件:settings.py

  1. #开启管道
  2.  
  3. ITEM_PIPELINES = {
  4.  
  5. 'secondblood.pipelines.SecondbloodPipeline': 300, #300表示为优先级,值越小优先级越高
  6.  
  7. }

2.1 基于mysql的管道存储

小试牛刀案例中,在管道文件里将item对象中的数据值存储到了磁盘中

,如果将item数据写入mysql数据库的话,只需要将上述案例中的管道文件修改成如下形式:

- pipelines.py文件

  1. # -*- coding: utf-8 -*-
  2.  
  3. # Define your item pipelines here
  4.  
  5. # Don't forget to add your pipeline to the ITEM_PIPELINES setting
  6.  
  7. # See: https://doc.scrapy.org/en/latest/topics/item-pipeline.html
  8.  
  9. #导入数据库的类
  10.  
  11. import pymysql
  12.  
  13. class QiubaiproPipelineByMysql(object):
  14.  
  15. conn = None #mysql的连接对象声明
  16.  
  17. cursor = None#mysql游标对象声明
  18.  
  19. def open_spider(self,spider):
  20.  
  21. print('开始爬虫')
  22.  
  23. #链接数据库
  24.  
  25. self.conn = pymysql.Connect(host='127.0.0.1',port=3306,user='root',password='123456',db='qiubai')
  26.  
  27. #编写向数据库中存储数据的相关代码
  28.  
  29. def process_item(self, item, spider):
  30.  
  31. #1.链接数据库
  32.  
  33. #2.执行sql语句
  34.  
  35. sql = 'insert into qiubai values("%s","%s")'%(item['author'],item['content'])
  36.  
  37. self.cursor = self.conn.cursor()
  38.  
  39. #执行事务
  40.  
  41. try:
  42.  
  43. self.cursor.execute(sql)
  44.  
  45. self.conn.commit()
  46.  
  47. except Exception as e:
  48.  
  49. print(e)
  50.  
  51. self.conn.rollback()
  52.  
  53. return item
  54.  
  55. def close_spider(self,spider):
  56.  
  57. print('爬虫结束')
  58.  
  59. self.cursor.close()
  60.  
  61. self.conn.close()

- settings.py

  1. ITEM_PIPELINES = {
  2.  
  3. 'qiubaiPro.pipelines.QiubaiproPipelineByMysql': 300,
  4.  
  5. }

2.2 基于redis的管道存储

小试牛刀案例中,在管道文件里将item对象中的数据值存储到了磁盘中,如果将item数据写入redis数据库的话,只需要将上述案例中的管道文件修改成如下形式:

  1. # -*- coding: utf-8 -*-
  2.  
  3. # Define your item pipelines here
  4.  
  5. # Don't forget to add your pipeline to the ITEM_PIPELINES setting
  6.  
  7. # See: https://doc.scrapy.org/en/latest/topics/item-pipeline.html
  8.  
  9. import redis
  10.  
  11. class QiubaiproPipelineByRedis(object):
  12.  
  13. conn = None
  14.  
  15. def open_spider(self,spider):
  16.  
  17. print('开始爬虫')
  18.  
  19. #创建链接对象
  20.  
  21. self.conn = redis.Redis(host='127.0.0.1',port=6379)
  22.  
  23. def process_item(self, item, spider):
  24.  
  25. dict = {
  26.  
  27. 'author':item['author'],
  28.  
  29. 'content':item['content']
  30.  
  31. }
  32.  
  33. #写入redis中
  34.  
  35. self.conn.lpush('data', dict)
  36.  
  37. return item

自己示例:

  1. import redis
  2.  
  3. class Qiubaipro1Pipeline(object):
  4. def __init__(self):
  5. self.conn = None
  6. self.key_data = 'dict_data4'
  7.  
  8. def open_spider(self, spider):
  9. print('start ................')
  10. self.conn = redis.Redis(host='127.0.0.1', port=6379, charset='utf-8', db=0, decode_responses=True)
  11. if self.conn.exists(self.key_data):
  12. self.conn.delete(self.key_data)
  13.  
  14. def process_item(self, item, spider):
  15. dict_data = {
  16. 'author': item['author'],
  17. 'content': item['content'],
  18. }
  19. print(dict_data)
  20.  
  21. self.conn.lpush(self.key_data, str(dict_data)) # python3.6 不能直接存字典进去
  22. return item
  23.  
  24. def close_spider(self, spider):
  25. print('end................')
  26. print(self.conn.lrange(self.key_data, 0, -1))
  27. self.conn.close()

- pipelines.py文件

  1. ITEM_PIPELINES = {
  2.  
  3. 'qiubaiPro.pipelines.QiubaiproPipelineByRedis': 300,
  4.  
  5. }

- 面试题:如果最终需要将爬取到的数据值一份存储到磁盘文件,一份存储到数据库中,则应该如何操作scrapy?

- 答:管道文件中的代码为

  1. #该类为管道类,该类中的process_item方法是用来实现持久化存储操作的。
  2.  
  3. class DoublekillPipeline(object):
  4.  
  5. def process_item(self, item, spider):
  6.  
  7. #持久化操作代码 (方式1:写入磁盘文件)
  8.  
  9. return item
  10.  
  11. #如果想实现另一种形式的持久化操作,则可以再定制一个管道类:
  12.  
  13. class DoublekillPipeline_db(object):
  14.  
  15. def process_item(self, item, spider):
  16.  
  17. #持久化操作代码 (方式1:写入数据库)
  18.  
  19. return item

在settings.py开启管道操作代码为:

  1. #下列结构为字典,字典中的键值表示的是即将被启用执行的管道文件和其执行的优先级。
  2.  
  3. ITEM_PIPELINES = {
  4.  
  5. 'doublekill.pipelines.DoublekillPipeline': 300,
  6.  
  7. 'doublekill.pipelines.DoublekillPipeline_db': 200,
  8.  
  9. }
  10.  
  11. #上述代码中,字典中的两组键值分别表示会执行管道文件中对应的两个管道类中的process_item方法,实现两种不同形式的持久化操作。

scrapy框架之持久化操作的更多相关文章

  1. (六--二)scrapy框架之持久化操作

    scrapy框架之持久化操作 基于终端指令的持久化存储 基于管道的持久化存储 1 基于终端指令的持久化存储 保证爬虫文件的parse方法中有可迭代类型对象(通常为列表or字典)的返回,该返回值可以通过 ...

  2. 爬虫开发8.scrapy框架之持久化操作

    今日概要 基于终端指令的持久化存储 基于管道的持久化存储 今日详情 1.基于终端指令的持久化存储 保证爬虫文件的parse方法中有可迭代类型对象(通常为列表or字典)的返回,该返回值可以通过终端指令的 ...

  3. scrapy框架之分布式操作

    分布式概念 分布式爬虫: 1.概念:多台机器上可以执行同一个爬虫程序,实现网站数据的分布爬取. 2.原生的scrapy是不可以实现分布式爬虫? a)调度器无法共享 b)管道无法共享 3.scrapy- ...

  4. 6 scrapy框架之分布式操作

    分布式爬虫 一.redis简单回顾 1.启动redis: mac/linux:   redis-server redis.conf windows: redis-server.exe redis-wi ...

  5. scrapy框架的持久化存储

    一 . 基于终端指令的持久化存储 保证爬虫文件的parse方法中有可迭代类型对象(通常为列表or字典)的返回,该返回值可以通过终端指令的形式写入指定格式的文件中进行持久化操作. 执行输出指定格式进行存 ...

  6. 爬虫开发14.scrapy框架之分布式操作

    分布式爬虫 一.redis简单回顾 1.启动redis: mac/linux:   redis-server redis.conf windows: redis-server.exe redis-wi ...

  7. Scrapy 框架,持久化文件相关

    持久化相关 相关文件 items.py 数据结构模板文件.定义数据属性. pipelines.py 管道文件.接收数据(items),进行持久化操作. 持久化流程 1.爬虫文件爬取到数据后,需要将数据 ...

  8. scrapy框架之CrawlSpider操作

    提问:如果想要通过爬虫程序去爬取”糗百“全站数据新闻数据的话,有几种实现方法? 方法一:基于Scrapy框架中的Spider的递归爬取进行实现(Request模块递归回调parse方法). 方法二:基 ...

  9. 爬虫开发11.scrapy框架之CrawlSpider操作

    提问:如果想要通过爬虫程序去爬取”糗百“全站数据新闻数据的话,有几种实现方法? 方法一:基于Scrapy框架中的Spider的递归爬取进行实现(Request模块递归回调parse方法). 方法二:基 ...

随机推荐

  1. for in和for of

  2. hadoop 常见 命令

    一  hadoop namenode 命令 1 格式化namanode 磁盘  hadoop namenode -format 二  hadoop fs 命令     和 linux  命令 非常类似 ...

  3. js判斷是否是正整數

    var curt_page = $("input[name='curt_page']").val(); if (!(/(^[1-9]\d*$)/.test(curt_page))) ...

  4. 使用spark streaming报错ERROR DFSClient: Failed to close inode xxxx

    转载自:http://blog.csdn.net/xiaolixiaoyi/article/details/45875101 好几个Spark streaming的程序同时运行,发现spark报出了如 ...

  5. 深入理解java虚拟机读后总结(个人总结记录)

    1.jvm布局:   jdk1.6版本JVM布局分为:heap(堆),method(方法区),stack(虚拟机栈),native stack(本地方法栈),程序计数器共五大区域. 其中方法区包含运行 ...

  6. [转] nginx配置优化+负载均衡+动静分离(附带参数解析)

    #指定nginx进程运行用户以及用户组user www www;#nginx要开启的进程数为8worker_processes  8;#全局错误日志文件#debug输出日志最为详细,而crit输出日志 ...

  7. TouchSlide 插件参数

    TouchSlide 可以说是 SuperSlide 手机简化版,不同的地方在于: 1.TouchSlide是纯javascript开发的,不依赖任何js库,鉴于此,TouchSlide调用方法和Su ...

  8. ROS学习材料/链接

    ROS的3D模型建立:http://blog.csdn.net/sujun3304/article/details/18962719 ubuntu14 ROS安装:http://blog.csdn.n ...

  9. html json 导出Excel

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  10. Linux bash笔记

    关于bash脚本,一般开头都加#!/bin/bash,表示以bash来作为脚本解释器:如果不加的话,就会默认当前用户登陆的shell为脚本解释器(很多情况下为sh,sh与bash不同,有可能导致脚本无 ...