Python爬虫 --- 2.2 Scrapy 选择器的介绍

在使用Scrapy框架之前,我们必须先了解它是如何筛选数据的,

Scrapy提取数据有自己的一套机制,被称作选择器(selectors),通过特定的Xpath或者CSS表达式来选择HTML文件的某个部分
Xpath是专门在XML文件中选择节点的语言,也可以用在HTML上。
CSS是一门将HTML文档样式化语言,选择器由它定义,并与特定的HTML元素的样式相关联。而且这些选择器构造于‘lxml’之上,这就意味着Scrapy框架下的数据筛选有着很高的效率。

基本选择器:

Scrapy爬虫支持多种信息提取的方法:

  • Beautiful Soup
  • Lxml
  • re
  • XPath Selector
  • CSS Selector

下面我们来介绍Xpath选择器和CSS选择器的使用:

Xpath选择器

  1. 介绍一下XPath:

    XPath 是一门在xml文档中查找信息的语言,它可以在XML文档中对于原色和属性进行遍历。其内置了超过100个内建函数,这些函数用于对字符串值,数值、日期、时间进行比较遍历。总之是一门很方便的语言。

    在网络爬虫中,我们只需要利用XPath来采集数据,所以只要掌握一些基本语法,就可以上手使用了。

  2. 基本使用语法,如下表:

    image
  1. 实例介绍:

    下面我们将以这个book.xml为例子来介绍:

    <html>
        <body>
            <bookstore>
                <book>
                    <title>水浒传</title>
                    <author>施耐庵</author>
                    <price>58.95</price>
                </book>
                <book>
                    <title>西游记</title>
                    <author>吴承恩</author>
                    <price>58.3</price>
                </book>
                <book>
                    <title>三国演义</title>
                    <author>罗贯中</author>
                    <price>48.3</price>
                </book>
                <book>
                    <title>红楼梦</title>
                    <author>曹雪芹</author>
                    <price>75</price>
                </book>
            </bookstore>
        </body>
    </html>
    
    • 先将我们需要使用的模块导入(调试环境为ipython):

      In [1]: from scrapy.selector import Selector
      
      In [2]: body = open('book.xml','r').read()
      
      In [3]: print(body)
      <html>
          <body>
              <bookstore>
                  <book>
                      <title>水浒传</title>
                      <author>施耐庵</author>
                      <price>58.95</price>
                  </book>
                  <book>
                      <title>西游记</title>
                      <author>吴承恩</author>
                      <price>58.3</price>
                  </book>
                  <book>
                      <title>三国演义</title>
                      <author>罗贯中</author>
                      <price>48.3</price>
                  </book>
                  <book>
                      <title>红楼梦</title>
                      <author>曹雪芹</author>
                      <price>75</price>
                  </book>
              </bookstore>
          </body>
      </html>
      
      In [4]: body
      Out[4]: '<html>\n\t<body>\n\t\t<bookstore>\n\t\t\t<book>\n\t\t\t\t<title>水浒传</title>\n\t\t\t\t<author>施耐庵</author>\n\t\t\t\t<price>58.95</price>\n\t\t\t</book>\n\t\t\t<book>\n\t\t\t\t<title>西游记</title>\n\t\t\t\t<author>吴承恩</author>\n\t\t\t\t<price>58.3</price>\n\t\t\t</book>\n\t\t\t<book>\n\t\t\t\t<title>三国演义</title>\n\t\t\t\t<author>罗贯中</author>\n\t\t\t\t<price>48.3</price>\n\t\t\t</book>\n\t\t\t<book>\n\t\t\t\t<title>红楼梦</title>\n\t\t\t\t<author>曹雪芹</author>\n\t\t\t\t<price>75</price>\n\t\t\t</book>\n\t\t</bookstore>\n\t</body>\n</html>'
      
      In [5]: 
      
    • 下面我们来举几个小例子,说明一下如何通过xpath找到我们想要的数据:

      In [5]: print("如果我们要第一个book的内容")
      如果我们要第一个book的内容
      
      In [7]: Selector(text=body).xpath('/html/body/bookstore/book[1]').extract()
      Out[7]: ['<book>\n\t\t\t\t<title>水浒传</title>\n\t\t\t\t<author>施耐庵</author>\n\t\t\t\t<price>58.95</price>\n\t\t\t</book>']
      
      In [8]: print("如果我们要最后一个book的内容")
      如果我们要最后一个book的内容
      
      In [9]: Selector(text=body).xpath('/html/body/bookstore/book[last()]').extract()
      Out[9]: ['<book>\n\t\t\t\t<title>红楼梦</title>\n\t\t\t\t<author>曹雪芹</author>\n\t\t\t\t<price>75</price>\n\t\t\t</book>']
      
      In [10]: print("如果我们要最后一个book的author属性的文本")
      如果我们要最后一个book的author属性的文本
      
      In [11]: Selector(text=body).xpath('/html/body/bookstore/book[last()]/author/text()').extract()
      Out[11]: ['曹雪芹']
      
      In [12]: print("下面是xpath的嵌套使用")
      下面是xpath的嵌套使用
      
      In [13]: subbody=Selector(text=body).xpath('/html/body/bookstore/book[3]').extract()
      
      In [14]: Selector(text=subbody[0]).xpath('//author/text()').extract()
      Out[14]: ['罗贯中']
      
      In [15]: Selector(text=subbody[0]).xpath('//book/author/text()').extract()
      Out[15]: ['罗贯中']
      
      In [16]: Selector(text=subbody[0]).xpath('//book/title/text()').extract()
      Out[16]: ['三国演义']
      

      在Xpath中最常用的方法大该就是这些了......

CSS选择器

  1. 介绍一下CSS:

    和Xpath选择器比起来,感觉CSS选择器容易一些,跟写.css时方法基本一样,就是在获取内容时和Xpath不同,这里需要注意一下。

  2. 基本使用语法,如下表:

    表达式 说明
    * 选择所有节点
    #container 选择id为container的节点
    .container 选择所有class包含container的节点
    li a 选取所有li 下所有a节点
    ul + p 选取ul后面的第一个p元素
    div#container > ul 选取id为container的div的第一个ul子元素
    ul ~p 选取与ul相邻的所有p元素
    a[title] 选取所有有title属性的a元素
    a[href="http://jobbole.com"] 选取所有href属性为http://jobbole.com 的a元素
    a[href*="jobbole"] 选取所有href属性值中包含jobbole的a元素
    a[href^="http"] 选取所有href属性值中以http开头的a元素
    a[href$=".jpg"] 选取所有href属性值中以.jpg结尾的a元素
    input[type=radio]:checked 选择选中的radio的元素
    div:not(#container) 选取所有id为非container 的div属性
    li:nth-child(3) 选取第三个li元素
    li:nth-child(2n) 选取第偶数个li元素
  3. 实例介绍:

    下面我们还是以这个book.xml为例子来介绍:

    • 上面xpath讲过如何导入模块了,下面我们来举几个小例子,说明一下如何通过css找到我们想要的数据:

      In [2]: print("如果我们要所有节点的内容")
      如果我们所有节点的内容
      
      In [3]: Selector(text=body).css('*').extract()
      Out[3]:
      ['<html>\n\t<body>\n\t\t<bookstore>\n\t\t\t<book>\n\t\t\t\t<title>水浒传</title>\n\t\t\t\t<author>施耐庵</author>\n\t\t\t\t<price>58.95</price>\n\t\t\t</book>\n\t\t\t<book>\n\t\t\t\t<title>西游记</title>\n\t\t\t\t<author>吴承恩</author>\n\t\t\t\t<price>58.3</price>\n\t\t\t</book>\n\t\t\t<book>\n\t\t\t\t<title>三国演义</title>\n\t\t\t\t<author>罗贯中</author>\n\t\t\t\t<price>48.3</price>\n\t\t\t</book>\n\t\t\t<book>\n\t\t\t\t<title>红楼梦</title>\n\t\t\t\t<author>曹雪芹</author>\n\t\t\t\t<price>75</price>\n\t\t\t</book>\n\t\t</bookstore>\n\t</body>\n</html>',
      '<body>\n\t\t<bookstore>\n\t\t\t<book>\n\t\t\t\t<title>水浒传</title>\n\t\t\t\t<author>施耐庵</author>\n\t\t\t\t<price>58.95</price>\n\t\t\t</book>\n\t\t\t<book>\n\t\t\t\t<title>西游记</title>\n\t\t\t\t<author>吴承恩</author>\n\t\t\t\t<price>58.3</price>\n\t\t\t</book>\n\t\t\t<book>\n\t\t\t\t<title>三国演义</title>\n\t\t\t\t<author>罗贯中</author>\n\t\t\t\t<price>48.3</price>\n\t\t\t</book>\n\t\t\t<book>\n\t\t\t\t<title>红楼梦</title>\n\t\t\t\t<author>曹雪芹</author>\n\t\t\t\t<price>75</price>\n\t\t\t</book>\n\t\t</bookstore>\n\t</body>',
      '<bookstore>\n\t\t\t<book>\n\t\t\t\t<title>水浒传</title>\n\t\t\t\t<author>施耐庵</author>\n\t\t\t\t<price>58.95</price>\n\t\t\t</book>\n\t\t\t<book>\n\t\t\t\t<title>西游记</title>\n\t\t\t\t<author>吴承恩</author>\n\t\t\t\t<price>58.3</price>\n\t\t\t</book>\n\t\t\t<book>\n\t\t\t\t<title>三国演义</title>\n\t\t\t\t<author>罗贯中</author>\n\t\t\t\t<price>48.3</price>\n\t\t\t</book>\n\t\t\t<book>\n\t\t\t\t<title>红楼梦</title>\n\t\t\t\t<author>曹雪芹</author>\n\t\t\t\t<price>75</price>\n\t\t\t</book>\n\t\t</bookstore>',
      '<book>\n\t\t\t\t<title>水浒传</title>\n\t\t\t\t<author>施耐庵</author>\n\t\t\t\t<price>58.95</price>\n\t\t\t</book>',
      '<title>水浒传</title>',
      '<author>施耐庵</author>',
      '<price>58.95</price>',
      '<book>\n\t\t\t\t<title>西游记</title>\n\t\t\t\t<author>吴承恩</author>\n\t\t\t\t<price>58.3</price>\n\t\t\t</book>',
      '<title>西游记</title>',
      '<author>吴承恩</author>',
      '<price>58.3</price>',
      '<book>\n\t\t\t\t<title>三国演义</title>\n\t\t\t\t<author>罗贯中</author>\n\t\t\t\t<price>48.3</price>\n\t\t\t</book>',
      '<title>三国演义</title>',
      '<author>罗贯中</author>',
      '<price>48.3</price>',
      '<book>\n\t\t\t\t<title>红楼梦</title>\n\t\t\t\t<author>曹雪芹</author>\n\t\t\t\t<price>75</price>\n\t\t\t</book>',
      '<title>红楼梦</title>',
      '<author>曹雪芹</author>',
      '<price>75</price>']
      
      In [4]: print("如果我们要bookstore下的所有内容")
      如果我们要bookstore下的所有内容
      
      In [5]: Selector(text=body).css('bookstore book').extract()
      Out[5]:
      ['<book>\n\t\t\t\t<title>水浒传</title>\n\t\t\t\t<author>施耐庵</author>\n\t\t\t\t<price>58.95</price>\n\t\t\t</book>',
      '<book>\n\t\t\t\t<title>西游记</title>\n\t\t\t\t<author>吴承恩</author>\n\t\t\t\t<price>58.3</price>\n\t\t\t</book>',
      '<book>\n\t\t\t\t<title>三国演义</title>\n\t\t\t\t<author>罗贯中</author>\n\t\t\t\t<price>48.3</price>\n\t\t\t</book>',
      '<book>\n\t\t\t\t<title>红楼梦</title>\n\t\t\t\t<author>曹雪芹</author>\n\t\t\t\t<price>75</price>\n\t\t\t</book>']
      

      由于book.xml没有元素,只有节点,所以只能列举以上例子,大家可以看到,css选择器比起xpath选择器更为的简洁。

好了,以上就是对Scrapy 选择器的介绍以及简单的使用,后面我会慢慢介绍Scrapy框架的具体使用。。。

此文章同时同步到我的个人博客http://www.fkomm.cn/ 谢谢支持

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

推荐阅读更多精彩内容