python爬虫入门教程--HTML文本的解析库BeautifulSoup(四)
发布时间 - 2026-01-11 01:17:19 点击率:次前言

python爬虫系列文章的第3篇介绍了网络请求库神器 Requests ,请求把数据返回来之后就要提取目标数据,不同的网站返回的内容通常有多种不同的格式,一种是 json 格式,这类数据对开发者来说最友好。另一种 XML 格式的,还有一种最常见格式的是 HTML 文档,今天就来讲讲如何从 HTML 中提取出感兴趣的数据
自己写个 HTML 解析器来解析吗?还是用正则表达式?这些都不是最好的办法,好在,Python 社区在这方便早就有了很成熟的方案,BeautifulSoup 就是这一类问题的克星,它专注于 HTML 文档操作,名字来源于 Lewis Carroll 的一首同名诗歌。
BeautifulSoup 是一个用于解析 HTML 文档的 Python 库,通过 BeautifulSoup,你只需要用很少的代码就可以提取出 HTML 中任何感兴趣的内容,此外,它还有一定的 HTML 容错能力,对于一个格式不完整的HTML 文档,它也可以正确处理。
安装 BeautifulSoup
pip install beautifulsoup4
BeautifulSoup3 被官方放弃维护,你要下载最新的版本 BeautifulSoup4。
HTML 标签
学习 BeautifulSoup4 前有必要先对 HTML 文档有一个基本认识,如下代码,HTML 是一个树形组织结构。
<html> <head> <title>hello, world</title> </head> <body> <h1>BeautifulSoup</h1> <p>如何使用BeautifulSoup</p> <body> </html>
- 它由很多标签(Tag)组成,比如 html、head、title等等都是标签
- 一个标签对构成一个节点,比如 ... 是一个根节点
- 节点之间存在某种关系,比如 h1 和 p 互为邻居,他们是相邻的兄弟(sibling)节点
- h1 是 body 的直接子(children)节点,还是 html 的子孙(descendants)节点
- body 是 p 的父(parent)节点,html 是 p 的祖辈(parents)节点
- 嵌套在标签之间的字符串是该节点下的一个特殊子节点,比如 “hello, world” 也是一个节点,只不过没名字。
使用 BeautifulSoup
构建一个 BeautifulSoup 对象需要两个参数,第一个参数是将要解析的 HTML 文本字符串,第二个参数告诉 BeautifulSoup 使用哪个解析器来解析 HTML。
解析器负责把 HTML 解析成相关的对象,而 BeautifulSoup 负责操作数据(增删改查)。”html.parser” 是Python内置的解析器,”lxml” 则是一个基于c语言开发的解析器,它的执行速度更快,不过它需要额外安装
通过 BeautifulSoup 对象就可以定位到 HTML 中的任何一个标签节点。
from bs4 import BeautifulSoup text = """ <html> <head> <title >hello, world</title> </head> <body> <h1>BeautifulSoup</h1> <p class="bold">如何使用BeautifulSoup</p> <p class="big" id="key1"> 第二个p标签</p> <a href="http://foofish.net" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" >python</a> </body> </html> """ soup = BeautifulSoup(text, "html.parser") # title 标签 >>> soup.title <title>hello, world</title> # p 标签 >>> soup.p <p class="bold">\u5982\u4f55\u4f7f\u7528BeautifulSoup</p> # p 标签的内容 >>> soup.p.string u'\u5982\u4f55\u4f7f\u7528BeautifulSoup'
BeatifulSoup 将 HTML 抽象成为 4 类主要的数据类型,分别是Tag , NavigableString , BeautifulSoup,Comment 。每个标签节点就是一个Tag对象,NavigableString 对象一般是包裹在Tag对象中的字符串,BeautifulSoup 对象代表整个 HTML 文档。例如:
>>> type(soup) <class 'bs4.BeautifulSoup'> >>> type(soup.h1) <class 'bs4.element.Tag'> >>> type(soup.p.string) <class 'bs4.element.NavigableString'>
Tag
每个 Tag 都有一个名字,它对应 HTML 的标签名称。
>>> soup.h1.name u'h1' >>> soup.p.name u'p'
标签还可以有属性,属性的访问方式和字典是类似的,它返回一个列表对象
>>> soup.p['class'] [u'bold']
NavigableString
获取标签中的内容,直接使用 .stirng 即可获取,它是一个 NavigableString 对象,你可以显式地将它转换为 unicode 字符串。
>>> soup.p.string u'\u5982\u4f55\u4f7f\u7528BeautifulSoup' >>> type(soup.p.string) <class 'bs4.element.NavigableString'> >>> unicode_str = unicode(soup.p.string) >>> unicode_str u'\u5982\u4f55\u4f7f\u7528BeautifulSoup'
基本概念介绍完,现在可以正式进入主题了,如何从 HTML 中找到我们关心的数据?BeautifulSoup 提供了两种方式,一种是遍历,另一种是搜索,通常两者结合来完成查找任务。
遍历文档树
遍历文档树,顾名思义,就是是从根节点 html 标签开始遍历,直到找到目标元素为止,遍历的一个缺陷是,如果你要找的内容在文档的末尾,那么它要遍历整个文档才能找到它,速度上就慢了。因此还需要配合第二种方法。
通过遍历文档树的方式获取标签节点可以直接通过 .标签名的方式获取,例如:
获取 body 标签:
>>> soup.body <body>\n<h1>BeautifulSoup</h1>\n<p class="bold">\u5982\u4f55\u4f7f\u7528BeautifulSoup</p>\n</body>
获取 p 标签
>>> soup.body.p <p class="bold">\u5982\u4f55\u4f7f\u7528BeautifulSoup</p>
获取 p 标签的内容
>>> soup.body.p.string \u5982\u4f55\u4f7f\u7528BeautifulSoup
前面说了,内容也是一个节点,这里就可以用 .string 的方式得到。遍历文档树的另一个缺点是只能获取到与之匹配的第一个子节点,例如,如果有两个相邻的 p 标签时,第二个标签就没法通过 .p 的方式获取,这是需要借用 next_sibling 属性获取相邻且在后面的节点。此外,还有很多不怎么常用的属性,比如:.contents 获取所有子节点,.parent 获取父节点,更多的参考请查看官方文档。
搜索文档树
搜索文档树是通过指定标签名来搜索元素,另外还可以通过指定标签的属性值来精确定位某个节点元素,最常用的两个方法就是 find 和 find_all。这两个方法在 BeatifulSoup 和 Tag 对象上都可以被调用。
find_all()
find_all( name , attrs , recursive , text , **kwargs )
find_all 的返回值是一个 Tag 组成的列表,方法调用非常灵活,所有的参数都是可选的。
第一个参数 name 是标签节点的名字。
# 找到所有标签名为title的节点
>>> soup.find_all("title")
[<title>hello, world</title>]
>>> soup.find_all("p")
[<p class="bold">\xc8\xe7\xba\xce\xca\xb9\xd3\xc3BeautifulSoup</p>,
<p class="big"> \xb5\xda\xb6\xfe\xb8\xf6p\xb1\xea\xc7\xa9</p>]
第二个参数是标签的class属性值
# 找到所有class属性为big的p标签
>>> soup.find_all("p", "big")
[<p class="big"> \xb5\xda\xb6\xfe\xb8\xf6p\xb1\xea\xc7\xa9</p>]
等效于
>>> soup.find_all("p", class_="big")
[<p class="big"> \xb5\xda\xb6\xfe\xb8\xf6p\xb1\xea\xc7\xa9</p>]
因为 class 是 Python 关键字,所以这里指定为 class_。
kwargs 是标签的属性名值对,例如:查找有href属性值为 "http://foofish.net" 的标签
>>> soup.find_all(href="http://foofish.net" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" ) [<a href="http://foofish.net" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" >python</a>]
当然,它还支持正则表达式
>>> import re
>>> soup.find_all(href=re.compile("^http"))
[<a href="http://foofish.net" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" >python</a>]
属性除了可以是具体的值、正则表达式之外,它还可以是一个布尔值(True/Flase),表示有属性或者没有该属性。
>>> soup.find_all(id="key1") [<p class="big" id="key1"> \xb5\xda\xb6\xfe\xb8\xf6p\xb1\xea\xc7\xa9</p>] >>> soup.find_all(id=True) [<p class="big" id="key1"> \xb5\xda\xb6\xfe\xb8\xf6p\xb1\xea\xc7\xa9</p>]
遍历和搜索相结合查找,先定位到 body 标签,缩小搜索范围,再从 body 中找 a 标签。
>>> body_tag = soup.body
>>> body_tag.find_all("a")
[<a href="http://foofish.net" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" >python</a>]
find()
find 方法跟 find_all 类似,唯一不同的地方是,它返回的单个 Tag 对象而非列表,如果没找到匹配的节点则返回 None。如果匹配多个 Tag,只返回第0个。
>>> body_tag.find("a")
<a href="http://foofish.net" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" >python</a>
>>> body_tag.find("p")
<p class="bold">\xc8\xe7\xba\xce\xca\xb9\xd3\xc3BeautifulSoup</p>
get_text()
获取标签里面内容,除了可以使用 .string 之外,还可以使用 get_text 方法,不同的地方在于前者返回的一个 NavigableString 对象,后者返回的是 unicode 类型的字符串。
>>> p1 = body_tag.find('p').get_text()
>>> type(p1)
<type 'unicode'>
>>> p1
u'\xc8\xe7\xba\xce\xca\xb9\xd3\xc3BeautifulSoup'
>>> p2 = body_tag.find("p").string
>>> type(p2)
<class 'bs4.element.NavigableString'>
>>> p2
u'\xc8\xe7\xba\xce\xca\xb9\xd3\xc3BeautifulSoup'
>>>
实际场景中我们一般使用 get_text 方法获取标签中的内容。
总结
BeatifulSoup 是一个用于操作 HTML 文档的 Python 库,初始化 BeatifulSoup 时,需要指定 HTML 文档字符串和具体的解析器。BeatifulSoup 有3类常用的数据类型,分别是 Tag、NavigableString、和 BeautifulSoup。查找 HTML元素有两种方式,分别是遍历文档树和搜索文档树,通常快速获取数据需要二者结合。
好了,以上就是关于这篇文章的全部内容,希望本文的内容对大家学习或者使用python能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持。
# python
# beautifulsoup
# 爬虫
# 文本
# Python正则获取、过滤或者替换HTML标签的方法
# python 解析html之BeautifulSoup
# Python使用lxml模块和Requests模块抓取HTML页面的教程
# Python使用BeautifulSoup库解析HTML基本使用教程
# Python HTML解析:BeautifulSoup
# Lxml
# XPath使用教程
# 文档
# 遍历
# 是一个
# 第二个
# 还可以
# 的是
# 都是
# 它还
# 第一个
# 感兴趣
# 如何使用
# 就可以
# 正则表达式
# 这是
# 都有
# 好了
# 都不
# 你可以
# 多个
# 你要
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
如何用低价快速搭建高质量网站?
微信小程序 scroll-view组件实现列表页实例代码
Laravel怎么导出Excel文件_Laravel Excel插件使用教程
韩国网站服务器搭建指南:VPS选购、域名解析与DNS配置推荐
Bootstrap整体框架之JavaScript插件架构
香港服务器网站生成指南:免费资源整合与高速稳定配置方案
昵图网官网入口 昵图网素材平台官方入口
Laravel PHP版本要求一览_Laravel各版本环境要求对照
网站优化排名时,需要考虑哪些问题呢?
胶州企业网站制作公司,青岛石头网络科技有限公司怎么样?
独立制作一个网站多少钱,建立网站需要花多少钱?
佛山企业网站制作公司有哪些,沟通100网上服务官网?
微信小程序制作网站有哪些,微信小程序需要做网站吗?
如何正确选择百度移动适配建站域名?
如何在搬瓦工VPS快速搭建网站?
Laravel如何使用查询构建器?(Query Builder高级用法)
如何实现javascript表单验证_正则表达式有哪些实用技巧
Laravel怎么实现支付功能_Laravel集成支付宝微信支付
香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化
Internet Explorer官网直接进入 IE浏览器在线体验版网址
微信小程序 配置文件详细介绍
Win11摄像头无法使用怎么办_Win11相机隐私权限开启教程【详解】
为什么php本地部署后css不生效_静态资源加载失败修复技巧【技巧】
如何在 Go 中优雅地映射具有动态字段的 JSON 对象到结构体
软银砸40亿美元收购DigitalBridge 强化AI资料中心布局
手机网站制作与建设方案,手机网站如何建设?
高端企业智能建站程序:SEO优化与响应式模板定制开发
Laravel如何实现多对多模型关联?(Eloquent教程)
如何构建满足综合性能需求的优质建站方案?
Laravel怎么多语言本地化设置_Laravel语言包翻译与Locale动态切换【手册】
大连网站制作公司哪家好一点,大连买房网站哪个好?
Python数据仓库与ETL构建实战_Airflow调度流程详解
JavaScript如何实现倒计时_时间函数如何精确控制
Laravel如何实现文件上传和存储?(本地与S3配置)
如何在阿里云高效完成企业建站全流程?
谷歌浏览器如何更改浏览器主题 Google Chrome主题设置教程
用v-html解决Vue.js渲染中html标签不被解析的问题
详解Huffman编码算法之Java实现
网站制作企业,网站的banner和导航栏是指什么?
如何解决hover在ie6中的兼容性问题
Laravel用户密码怎么加密_Laravel Hash门面使用教程
青岛网站建设如何选择本地服务器?
头像制作网站在线观看,除了站酷,还有哪些比较好的设计网站?
个人网站制作流程图片大全,个人网站如何注销?
历史网站制作软件,华为如何找回被删除的网站?
如何破解联通资金短缺导致的基站建设难题?
如何用VPS主机快速搭建个人网站?
ChatGPT 4.0官网入口地址 ChatGPT在线体验官网
JavaScript如何操作视频_媒体API怎么控制播放
如何快速查询网站的真实建站时间?

