Python爬虫系列(四):Beautiful Soup解析HTML之把HTML转成Python对象

在前几篇文章,我们学会了如何获取html文档内容,就是从url下载网页。今天开始,我们将讨论如何将html转成python对象,用python代码对文档进行分析。

(牛小妹在学校折腾了好几天,也没把html文档给分析出来。接下来的几篇文章,你可就要好好看了)

Beautiful Soup将复杂HTML文档转换成一个复杂的树形结构,每个节点都是Python对象,所有对象可以归纳为4种: Tag , NavigableString , BeautifulSoup , Comment

Tag 对象与XML或HTML原生文档中的tag相同

获取和修改对象的名称及属性

from bs4 import BeautifulSoup

#注意,第二个参数一定是这样用字符串,照官方文档要报错。现在BeautifulSoup 是4.6

soup = BeautifulSoup('Extremely bold', "lxml-xml")

tag = soup.b

#b标签对应的python对象

print(type(tag))

print(tag.name)

#修改标签 name 不是标签的name属性,而是标签自身

tag.name = "blockquote"

print(tag)

#获取属性

print(tag['class'])

#获取多个属性

print(tag.attrs)

#修改属性

tag['class'] = 'verybold'

tag['id'] = 1

print(tag)

#删除属性

del tag['class']

del tag['id']

print(tag)

#已经没有class属性 获取就报错

print(tag['class'])

print(tag.get('class'))

多指属性:

是指一个属性有多个值。

注意:这里使用的是 lxml-xml 解析器 所以看不出来是多值。用html。parser转出来的就是多值。

css_soup = BeautifulSoup('', "lxml-xml")

print(css_soup.p['class'])

css_soup = BeautifulSoup('', "lxml-xml")

print(css_soup.p['class'])

对应结果:

body strikeout

body

如果某个属性看起来好像有多个值,但在任何版本的HTML定义中都没有被定义为多值属性,那么Beautiful Soup会将这个属性作为字符串返回

id_soup = BeautifulSoup('', "lxml-xml")

#返回的是字符串

print(id_soup.p['id'])

将tag转换成字符串时,多值属性会合并为一个值

rel_soup = BeautifulSoup('

Back to the homepage

', "lxml-xml")

print(rel_soup.a['rel'])

rel_soup.a['rel'] = ['index', 'contents']

print(rel_soup.p)

展示结果:

Back to the homepage

注意看a 标签的rel属性

可以遍历的字符串

字符串常被包含在tag内.Beautiful Soup用 NavigableString 类来包装tag中的字符串

soup = BeautifulSoup('Extremely bold')

tag = soup.b

print(tag.string)

print(type(tag.string))

结果:

Extremely bold

一个 NavigableString 字符串与Python中的Unicode字符串相同,并且还支持包含在遍历文档树搜索文档树中的一些特性. 通过 unicode() 方法可以直接将 NavigableString 对象转换成Unicode字符串

from bs4 import BeautifulSoup

from lxml.html.clean import unicode

soup = BeautifulSoup('Extremely bold')

tag = soup.b

unicode_string = unicode(tag.string)

print(unicode_string)

结果:

Extremely bold

tag中包含的字符串不能编辑,但是可以被替换成其它的字符串,用replace_with()方法:

from bs4 import BeautifulSoup

soup = BeautifulSoup('Extremely bold')

tag = soup.b

tag.string.replace_with("No longer bold")

print(tag)

结果:

No longer bold

注意:

NavigableString 对象支持遍历文档树搜索文档树中定义的大部分属性, 并非全部.尤其是,一个字符串不能包含其它内容(tag能够包含字符串或是其它tag),字符串不支持 .contents 或 .string 属性或 find() 方法.

如果想在Beautiful Soup之外使用 NavigableString 对象,需要调用 unicode() 方法,将该对象转换成普通的Unicode字符串,否则就算Beautiful Soup已方法已经执行结束,该对象的输出也会带有对象的引用地址.这样会浪费内存.

BeautifulSoup对象

BeautifulSoup 对象表示的是一个文档的全部内容.大部分时候,可以把它当作 Tag 对象,它支持遍历文档树搜索文档树中描述的大部分的方法.

因为 BeautifulSoup 对象并不是真正的HTML或XML的tag,所以它没有name和attribute属性.但有时查看它的 .name 属性是很方便的,所以 BeautifulSoup 对象包含了一个值为 “[document]” 的特殊属性 .name

知道即可

注释及特殊字符串

Tag , NavigableString , BeautifulSoup 几乎覆盖了html和xml中的所有内容,但是还有一些特殊对象.容易让人担心的内容是文档的注释部分

from bs4 import BeautifulSoup, CData

markup = ""

soup = BeautifulSoup(markup)

comment = soup.b.string

print(type(comment))

# Comment 对象是一个特殊类型的 NavigableString 对象:

print(comment)

#美化后的输出结果

print(soup.b.prettify())

# Beautiful Soup中定义的其它类型都可能会出现在XML的文档中:

# CData , ProcessingInstruction , Declaration , Doctype .与 Comment 对象类似,

# 这些类都是 NavigableString 的子类,只是添加了一些额外的方法的字符串独享.

# 下面是用CDATA来替代注释的例子:

cdata = CData("A CDATA block")

comment.replace_with(cdata)

print(soup.b.prettify())

#

#

#

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

推荐阅读更多精彩内容