设为首页收藏本站

Crossin的编程教室

 找回密码
 立即加入
查看: 5040|回复: 0
打印 上一主题 下一主题

BeautifulSoup:干了这碗“美丽汤”,网页解析倍儿爽

[复制链接]

169

主题

1

好友

733

积分

版主

Rank: 7Rank: 7Rank: 7

跳转到指定楼层
楼主
发表于 2019-3-6 13:14:44 |只看该作者 |倒序浏览

关于爬虫的案例和方法,我们已讲过许多。不过在以往的文章中,大多是关注在如何把网页上的内容抓取下来。今天我们来分享下,当你已经把内容爬下来之后,如何提取出其中你需要的具体信息。

网页被抓取下来,通常就是str 字符串类型的对象,要从里面寻找信息,最直接的想法就是直接通过字符串的 find 方法和切片操作:
  1. s = '<p>价格:15.7 元</p>'
  2. start = s.find('价格:')
  3. end = s.find(' 元')
  4. print(s[start+3:end])  
  5. # 15.7
复制代码
这能应付一些极简单的情况,但只要稍稍复杂一点,这么写就会累死人。更通用的做法是使用正则表达式:
  1. import re
  2. s = '<p>价格:15.7 元</p>'
  3. r = re.search('[\d.]+', s)
  4. print(r.group())
  5. # 15.7
复制代码
正则表达式是处理文本解析的万金油,什么情况都可以应对。但可惜掌握它需要一定的学习成本,原本我们有一个网页提取的问题,用了正则表达式,现在我们有了两个问题。

HTML 文档本身是结构化的文本,有一定的规则,通过它的结构可以简化信息提取。于是,就有了lxml、pyquery、BeautifulSoup等网页信息提取库。一般我们会用这些库来提取网页信息。其中,lxml 有很高的解析效率,支持 xPath 语法(一种可以在 HTML 中查找信息的规则语法);pyquery 得名于 jQuery(知名的前端 js 库),可以用类似 jQuery 的语法解析网页。但我们今天要说的,是剩下的这个:
BeautifulSoup
BeautifulSoup(下文简称 bs)翻译成中文就是“美丽的汤”,这个奇特的名字来源于《爱丽丝梦游仙境》(这也是为何在其官网会配上奇怪的插图,以及用《爱丽丝》的片段作为测试文本)。

bs 最大的特点我觉得是简单易用,不像正则和 xPath 需要刻意去记住很多特定语法,尽管那样会效率更高更直接。对大多数 python 使用者来说,好用会比高效更重要。这也是我自己使用并推荐 bs 的主要原因。

接下来介绍点 bs 的基本方法,让你看完就能用起来。考虑到“只收藏不看党”的阅读体验,先给出一个“嫌长不看版”的总结:
随anaconda附带,也可以通过pip安装指定不同解析器在性能、容错性上会有差异,导致结果也可能不一样基本使用流程:通过文本初始化 bs 对象->通过 find/find_all 或其他方法检测信息->输出或保存可以迭代式的查找,比如先定位出一段内容,再其上继续检索开发时应注意不同方法的返回类型,出错时多看报错、多加输出信息官方文档很友好,也有中文,推荐阅读安装
推荐使用pip进行安装(关于 pip 见前文《Crossin:如何安装 Python 的第三方模块》):
pip install beautifulsoup4
要注意,包名是beautifulsoup4,如果不加上 4,会是老版本也就是 bs3,它是为了兼容性而存在,目前已不推荐。我们这里说 bs,都是指 bs4。

bs4 也可以直接通过安装 anaconda 获得(介绍见前文《Crossin:Python数据科学环境:Anaconda 了解一下》)。

bs 在使用时需要指定一个“解析器”:
html.parse- python 自带,但容错性不够高,对于一些写得不太规范的网页会丢失部分内容lxml- 解析速度快,需额外安装xml- 同属 lxml 库,支持 XML 文档html5lib- 最好的容错性,但速度稍慢
这里的 lxml 和 html5lib 都需要额外安装,不过如果你用的是 anaconda,都是一并安装好的。
快速上手
我们就用官网上的文档作例子:
  1. html_doc = """
  2. <html><head><title>The Dormouse's story</title></head>
  3. <body>
  4. <p class="title"><b>The Dormouse's story</b></p>
  5. <p class="story">Once upon a time there were three little sisters; and their names were
  6. <a href="http://example.com/elsie" class="sister" id="link1">Elsie</a>,
  7. <a href="http://example.com/lacie" class="sister" id="link2">Lacie</a> and
  8. <a href="http://example.com/tillie" class="sister" id="link3">Tillie</a>;
  9. and they lived at the bottom of a well.</p>
  10. <p class="story">...</p>
  11. """
复制代码
使用 bs 的初始化操作,是用文本创建一个 BeautifulSoup 对象,建议手动指定解析器:
  1. from bs4 import BeautifulSoup
  2. soup = BeautifulSoup(html_doc, 'html.parser')
复制代码
获取其中的某个结构化元素及其属性:
  1. soup.title  # title 元素
  2. # <title>The Dormouse's story</title>
  3. soup.p  # 第一个 p 元素
  4. # <p class="title"><b>The Dormouse's story</b></p>
  5. soup.p['class']  # p 元素的 class 属性
  6. # ['title']
  7. soup.p.b  # p 元素下的 b 元素
  8. # <b>The Dormouse's story</b>
  9. soup.p.parent.name  # p 元素的父节点的标签
  10. # body
复制代码
并不是所有信息都可以简单地通过结构化获取,通常使用 find 和 find_all 方法进行查找:
  1. soup.find_all('a')  # 所有 a 元素
  2. # [<a class="sister" href="http://example.com/elsie" id="link1">Elsie</a>,
  3. #  <a class="sister" href="http://example.com/lacie" id="link2">Lacie</a>,
  4. #  <a class="sister" href="http://example.com/tillie" id="link3">Tillie</a>]
  5. soup.find(id='link3')  # id 为 link3 的元素
  6. # <a class="sister" href="http://example.com/tillie" id="link3">Tillie</a
复制代码
find 和 find_all 可以有多个搜索条件叠加,比如find('a', id='link3', class_='sister')find 返回的是一个bs4.element.Tag 对象,这个对象可以进一步进行搜索。如果有多个满足的结果,find只返回第一个;如果没有,返回 None。find_all 返回的是一个由 bs4.element.Tag 对象组成的 list,不管找到几个或是没找到,都是 list。
输出:
  1. x = soup.find(class_='story')
  2. x.get_text()  # 仅可见文本内容
  3. # 'Once upon a time there were three little sisters; and their names were\nElsie,\nLacie and\nTillie;\nand they lived at the bottom of a well.'
  4. x.prettify()  # 元素完整内容
  5. # '<p class="story">\n Once upon a time there were three little sisters; and their names were\n <a class="sister" href="http://example.com/elsie" id="link1">\n  Elsie\n </a>\n ,\n <a class="sister" href="http://example.com/lacie" id="link2">\n  Lacie\n </a>\n and\n <a class="sister" href="http://example.com/tillie" id="link3">\n  Tillie\n </a>\n ;\nand they lived at the bottom of a well.\n</p>\n'
复制代码
如果你有前端开发经验,对 CSS 选择器很熟悉,bs 也为你提供了相应的方法:
  1. soup.select('html head title')
  2. # [<title>The Dormouse's story</title>]
  3. soup.select('p > #link1')
  4. # [<a class="sister" href="http://example.com/elsie" id="link1">Elsie</a>]
复制代码
以上就是 BeautifulSoup 的一个极简上手介绍,对于 bs 能做什么,想必你已有了一个初步认识。如果你要在开发中使用,建议再看下它的官方文档。文档写得很清楚,也有中文版,你只要看了最初的一小部分,就可以在代码中派上用场了。更多的细节可以在使用时进一步搜索具体方法和参数设置。

中文版文档地址:
Beautiful Soup 4.2.0 文档www.crummy.com
对于爬虫的其他方面,推荐阅读我们之前的相关文章:
Chrome开发者工具:爬虫必备,掌握它就解决了一半的问题requests:让你的爬虫开发效率提升8倍goose 简介:一鹅在手,抓遍全球IP代理池:听说你好不容易写了个爬虫,结果没抓几个就被封了?selenium - 祖传爬虫利器Python爬虫:一些常用的爬虫技巧总结爬虫+网站开发实例:电影票比价网
════

其他文章及回答:

学编程:如何自学Python | 新手引导 | 一图学Python

开发案例:智能防挡弹幕 | 红包提醒 | 流浪地球

欢迎搜索及关注:Crossin的编程教室

回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即加入

QQ|手机版|Archiver|Crossin的编程教室 ( 苏ICP备15063769号  

GMT+8, 2024-11-22 19:54 , Processed in 0.014675 second(s), 21 queries .

Powered by Discuz! X2.5

© 2001-2012 Comsenz Inc.

回顶部