low-level~python scrapy多级页面爬取并存储为JSON格式

在上一篇scrapy(low-level~python scrapy自动爬取网页的爬虫)[https://www.jianshu.com/p/9b07e556216e]中我们实现了翻页操作,但是这种操作不利于改动。这次改进为分模块编程的思想。
思路:
第一步:提取每页的链接
第二步:提取每页商品的链接
第三步:提取每页商品的具体信息
这里的难点在于

for i in range(1,3):
     url ="http://category.dangdang.com/pg"+str(i)+"-cp01.05.16.00.00.00.html
      #通过yield返回Reques,并指定要爬取的网址和回调函数
      yield Request(url,callback=self.parse)

Request函数的参数callback,这个参数决定着接下来执行什么操作。
首先我们知道spider中初始的request是通过调用start_requests()来获取,start_requests()读取start_urls中的URL,所以我们每页链接的提取放在start_requests()中处理

def start_requests(self):
         start_urls = ['http://category.dangdang.com/pg1-cp01.05.16.00.00.00.html',]
          start,end=1,3
          for i in range(start,end):
              print(str(i))
              url="http://category.dangdang.com/pg"+str(i)+"-cp01.05.16.00.00.00.html"
              print(url)
              yield scrapy.http.Request(url,self.parse)

yield scrapy.http.Request(url,self.parse)含义:

 Request请求url链接,回调parse()函数

接下来我们定义函数parse(),这个函数负责提取每个页面商品的链接。分析页面代码


image.png

如图所示,所有的商品信息都在<li>*</li>标签下,而这些li标签在<ul class="bigimg" id="component_0__0__6612">下面。所以当我们得到页面的url之后,先提取ul class....这个大标签,然后循环提取每一个商品的链接标签,代码如下:

 def parse(self,response):
         urls = response.xpath("//*[@id='component_0__0__6612']/li")
         for url in urls:
              href=url.xpath("a[@class='pic']/@href").extract_first()
             print(href)
              request=scrapy.http.Request(href,callback=self.parseArticle)
             yield request

最后一步,进入商品详细链接中提取title属性
首先定义一个item,存储爬取的数据信息,然后用xpath提取我们需要的标签

def parseArticle(self,response):
          item =AutopjtItem()
          item['name']=response.xpath('//
           [@id="product_info"]/div[1]/h2/span[1]/@title').extract()
          print(item['name'])
          yield item

完成,中间遇到很多的坑,一一列举,避免再次发生
1,class类中的name值必须是我们的文件名,否则会报错:

KeyError: 'Spider not found: autospd

2,当我们提取<ul class="bigimg" id="component_0__0__6612">大标签时,不能用extract()函数。否则什么都得不到(不要问我怎么知道的,都是泪,,,,,,,)


image.png

完整代码如下

# -*- coding: utf-8 -*-
  2 import scrapy
  3 from autopjt.items import AutopjtItem
  4 from scrapy.http import Request
  5 from scrapy.selector import Selector
  6 #创建一个爬虫类AutospdSpider,该类继承了scrapy.Spider基类
  7 class AutospdSpider(scrapy.Spider):
  8     name = "autospd"
  9    # urlList = []
 10     #name属性代表的是爬虫名称
 11     #allowed_domains属性代表的是允许爬行的域名
 12     allowed_domains = ["dangdang.com"]
 13     #爬行的起始网址
 14    # start_urls = ['http://category.dangdang.com/pg1-cp01.05.16.00.00.00.html',]
 15     def start_requests(self):
 16         start_urls = ['http://category.dangdang.com/pg1-cp01.05.16.00.00.00.html',]
 17         start,end=1,3
 18         for i in range(start,end):
 19             print(str(i))
 20             url="http://category.dangdang.com/pg"+str(i)+"-cp01.05.16.00.00.00.html"
 21             print(url)
 22             yield scrapy.http.Request(url,self.parse)
 23 #得到页链接,对每个页链接找到每一个商品链接
 24     def parse(self,response):
 25         urls = response.xpath("//*[@id='component_0__0__6612']/li")
 26         for url in urls:
 27             href=url.xpath("a[@class='pic']/@href").extract_first()
 28             print(href)
 29             request=scrapy.http.Request(href,callback=self.parseArticle)
 30             yield request
 31     def parseArticle(self,response):
 32         item =AutopjtItem()
 33         item['name']=response.xpath('//*[@id="product_info"]/div[1]/h2/span[1]/@title').extract()
 34         yield item

当item在Spider中被收集之后,将会被传递到item Pipeline,Item Pipeline,在pipeline.py文件中对提取到的数据进行进一步的处理。

class AutopjtPipeline(object):
      def __init__(self):
          self.file = codecs.open("mydata.json","wb",encoding="utf-8")
       #这个方法必须返回一个Item对象,参数item:被爬取的item,spider爬取该item的spider
      def process_item(self, item, spider):
         #打开JSON文件,向里面以dumps的方式吸入数据,其中ensure_ascii=False,   不然数据会直接为utf编码的方式存入
          i = json.dumps(dict(item),ensure_ascii=False)
          #每条数据后加上换行
          line = i +'\n'
          #数据写到mydata.json文件中
          self.file.write(line)
         return item
      def close_spider(self,spider):
          self.file.close()

保存的文件的打开路径直接写想保存的XXX.json即可
再有一步就可以完成了,在pipelines.py处理成为json格式时,还需要在settings.py文件告诉系统pipelines文件在哪里以及pipeline文件里面对应的类是什么,找到settings.py文件中关于pipelines的设置部分,

ITEM_PIPELINES={
          'autopjt.pipelines.AutopjtPipeline':300,
    #分配给每个类的整型值,确定他们的运行顺序,item按数字从低到高的顺序通过pipeline
         }

在上面代码中,'autopjt.pipelines.AutopjtPipeline'中的autopjt为项目名(即Scrapy项目的核心目录名),pipelines代表autopjt目录下的pipelines.py文件的文件名,AutopjtPipeline代表对应的pipelines文件里的类。

再次运行spider文件,vim XXXX.json就可以打开json格式的文件


image.png

参考来源

狸狸深深【补充更新Report B2】Scrapy分页爬取四川大学公共管理学院全职教师信息及学院新闻
Requests and Responses

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

推荐阅读更多精彩内容