python爬蟲學(xué)習(xí)筆記之Beautifulsoup模塊用法詳解
本文實(shí)例講述了python爬蟲學(xué)習(xí)筆記之Beautifulsoup模塊用法。分享給大家供大家參考,具體如下:
相關(guān)內(nèi)容:
- 什么是beautifulsoup
- bs4的使用
- 導(dǎo)入模塊
- 選擇使用解析器
- 使用標(biāo)簽名查找
- 使用find\find_all查找
- 使用select查找
首發(fā)時(shí)間:2018-03-02 00:10
什么是beautifulsoup:
- 是一個(gè)可以從HTML或XML文件中提取數(shù)據(jù)的Python庫.它能夠通過你喜歡的轉(zhuǎn)換器實(shí)現(xiàn)慣用的文檔導(dǎo)航,查找,修改文檔的方式.(官方)
- beautifulsoup是一個(gè)解析器,可以特定的解析出內(nèi)容,省去了我們編寫正則表達(dá)式的麻煩。
Beautiful Soup 3 目前已經(jīng)停止開發(fā),我們推薦在現(xiàn)在的項(xiàng)目中使用Beautiful Soup 4
beautifulsoup的版本:最新版是bs4
bs4的使用:
1.導(dǎo)入模塊:
from bs4 import beautifulsoup
2.選擇解析器解析指定內(nèi)容:
soup=beautifulsoup(解析內(nèi)容,解析器)
常用解析器:html.parser,lxml,xml,html5lib
有時(shí)候需要安裝安裝解析器:比如pip3 install lxml
BeautifulSoup默認(rèn)支持Python的標(biāo)準(zhǔn)HTML解析庫,但是它也支持一些第三方的解析庫:
解析器之間的區(qū)別 #此處摘自官方文檔
Beautiful Soup為不同的解析器提供了相同的接口,但解析器本身時(shí)有區(qū)別的.同一篇文檔被不同的解析器解析后可能會(huì)生成不同結(jié)構(gòu)的樹型文檔.區(qū)別最大的是HTML解析器和XML解析器,看下面片段被解析成HTML結(jié)構(gòu):
BeautifulSoup("<a><b /></a>") # <html><head></head><body><a><b></b></a></body></html>因?yàn)榭諛?biāo)簽<b />不符合HTML標(biāo)準(zhǔn),所以解析器把它解析成<b></b>
同樣的文檔使用XML解析如下(解析XML需要安裝lxml庫).注意,空標(biāo)簽<b />依然被保留,并且文檔前添加了XML頭,而不是被包含在<html>標(biāo)簽內(nèi):
BeautifulSoup("<a><b /></a>", "xml") # <?xml version="1.0" encoding="utf-8"?> # <a><b/></a>HTML解析器之間也有區(qū)別,如果被解析的HTML文檔是標(biāo)準(zhǔn)格式,那么解析器之間沒有任何差別,只是解析速度不同,結(jié)果都會(huì)返回正確的文檔樹.
但是如果被解析文檔不是標(biāo)準(zhǔn)格式,那么不同的解析器返回結(jié)果可能不同.下面例子中,使用lxml解析錯(cuò)誤格式的文檔,結(jié)果</p>標(biāo)簽被直接忽略掉了:
BeautifulSoup("<a></p>", "lxml") # <html><body><a></a></body></html>使用html5lib庫解析相同文檔會(huì)得到不同的結(jié)果:
BeautifulSoup("<a></p>", "html5lib") # <html><head></head><body><a><p></p></a></body></html>html5lib庫沒有忽略掉</p>標(biāo)簽,而是自動(dòng)補(bǔ)全了標(biāo)簽,還給文檔樹添加了<head>標(biāo)簽.
使用pyhton內(nèi)置庫解析結(jié)果如下:
BeautifulSoup("<a></p>", "html.parser") # <a></a>與lxml [7] 庫類似的,Python內(nèi)置庫忽略掉了</p>標(biāo)簽,與html5lib庫不同的是標(biāo)準(zhǔn)庫沒有嘗試創(chuàng)建符合標(biāo)準(zhǔn)的文檔格式或?qū)⑽臋n片段包含在<body>標(biāo)簽內(nèi),與lxml不同的是標(biāo)準(zhǔn)庫甚至連<html>標(biāo)簽都沒有嘗試去添加.
因?yàn)槲臋n片段“<a></p>”是錯(cuò)誤格式,所以以上解析方式都能算作”正確”,html5lib庫使用的是HTML5的部分標(biāo)準(zhǔn),所以最接近”正確”.不過所有解析器的結(jié)構(gòu)都能夠被認(rèn)為是”正?!钡?
不同的解析器可能影響代碼執(zhí)行結(jié)果,如果在分發(fā)給別人的代碼中使用了 BeautifulSoup ,那么最好注明使用了哪種解析器,以減少不必要的麻煩.
3.操作【約定soup是beautifulsoup(解析內(nèi)容,解析器)返回的解析對(duì)象】:
-
使用標(biāo)簽名查找
- 使用標(biāo)簽名來獲取結(jié)點(diǎn):
- soup.標(biāo)簽名
- 使用標(biāo)簽名來獲取結(jié)點(diǎn)標(biāo)簽名【這個(gè)重點(diǎn)是name,主要用于非標(biāo)簽名式篩選時(shí),獲取結(jié)果的標(biāo)簽名】:
- soup.標(biāo)簽.name
- 使用標(biāo)簽名來獲取結(jié)點(diǎn)屬性:
- soup.標(biāo)簽.attrs【獲取全部屬性】
- soup.標(biāo)簽.attrs[屬性名]【獲取指定屬性】
- soup.標(biāo)簽[屬性名]【獲取指定屬性】
- soup.標(biāo)簽.get(屬性名)
- 使用標(biāo)簽名來獲取結(jié)點(diǎn)的文本內(nèi)容:
- soup.標(biāo)簽.text
- soup.標(biāo)簽.string
- soup.標(biāo)簽.get_text()
- 使用標(biāo)簽名來獲取結(jié)點(diǎn):
補(bǔ)充1:上面的篩選方式可以使用嵌套:
print(soup.p.a)#p標(biāo)簽下的a標(biāo)簽
補(bǔ)充2:以上的name,text,string,attrs等方法都可以使用在當(dāng)結(jié)果是一個(gè)bs4.element.Tag對(duì)象的時(shí)候:
from bs4 import BeautifulSoup html = """ <html > <head> <meta charset="UTF-8"> <title>this is a title</title> </head> <body> <p class="news">123</p> <p class="contents" id="i1">456</p> <a rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" >advertisements</a> </body> </html> """ soup = BeautifulSoup(html,'lxml') print("獲取結(jié)點(diǎn)".center(50,'-')) print(soup.head)#獲取head標(biāo)簽 print(soup.p)#返回第一個(gè)p標(biāo)簽 #獲取結(jié)點(diǎn)名 print("獲取結(jié)點(diǎn)名".center(50,'-')) print(soup.head.name) print(soup.find(id='i1').name) #獲取文本內(nèi)容 print("獲取文本內(nèi)容".center(50,'-')) print(soup.title.string)#返回title的內(nèi)容 print(soup.title.text)#返回title的內(nèi)容 print(soup.title.get_text()) #獲取屬性 print("-----獲取屬性-----") print(soup.p.attrs)#以字典形式返回標(biāo)簽的內(nèi)容 print(soup.p.attrs['class'])#以列表形式返回標(biāo)簽的值 print(soup.p['class'])#以列表形式返回標(biāo)簽的值 print(soup.p.get('class')) ############# t=soup.title print(type(t))#<class 'bs4.element.Tag'> print(t.name)#title print(t.text) #嵌套選擇: print(soup.head.title.string)
- 獲取子結(jié)點(diǎn)【直接獲取也會(huì)獲取到'\n',會(huì)認(rèn)為'\n'也是一個(gè)標(biāo)簽】:
- soup.標(biāo)簽.contents【返回值是一個(gè)列表】
- soup.標(biāo)簽.children【返回值是一個(gè)可迭代對(duì)象,獲取實(shí)際子結(jié)點(diǎn)需要迭代】
- 獲取子孫結(jié)點(diǎn):
- soup.標(biāo)簽.descendants【返回值也是一個(gè)可迭代對(duì)象,實(shí)際子結(jié)點(diǎn)需要迭代】
- 獲取父結(jié)點(diǎn):
- soup.標(biāo)簽.parent
- 獲取祖先結(jié)點(diǎn)[父結(jié)點(diǎn),祖父結(jié)點(diǎn),曾祖父結(jié)點(diǎn)…]:
- soup.標(biāo)簽.parents【】
- 獲取兄弟結(jié)點(diǎn):
- soup.next_sibling【獲取后面的一個(gè)兄弟結(jié)點(diǎn)】
- soup.next_siblings【獲取后面所有的兄弟結(jié)點(diǎn)】【返回值是一個(gè)可迭代對(duì)象】
- soup.previous_sibling【獲取前一兄弟結(jié)點(diǎn)】
- soup.previous_siblings【獲取前面所有的兄弟結(jié)點(diǎn)】【返回值是一個(gè)可迭代對(duì)象】
補(bǔ)充3:與補(bǔ)充2一樣,上面的函數(shù)都可以使用在當(dāng)結(jié)果是一個(gè)bs4.element.Tag對(duì)象的時(shí)候。
from bs4 import BeautifulSoup html = """ <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <p class="news"><a >123456</a> <a >78910</a> </p><p class="contents" id="i1"></p> <a rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" >advertisements</a> <span>aspan</span> </body> </html> """ soup = BeautifulSoup(html, 'lxml') #獲取子結(jié)點(diǎn) print("獲取子結(jié)點(diǎn)".center(50,'-')) print(soup.p.contents) print("\n") c=soup.p.children#返回的是一個(gè)可迭代對(duì)象 for i,child in enumerate(c): print(i,child) print("獲取子孫結(jié)點(diǎn)".center(50,'-')) print(soup.p.descendants) c2=soup.p.descendants for i,child in enumerate(c2): print(i,child) print("獲取父結(jié)點(diǎn)".center(50,'-')) c3=soup.title.parent print(c3) print("獲取父,祖先結(jié)點(diǎn)".center(50,'-')) c4=soup.title.parents print(c4) for i,child in enumerate(c4): print(i,child) print("獲取兄弟結(jié)點(diǎn)".center(50,'-')) print(soup.p.next_sibling) print(soup.p.previous_sibling) for i,child in enumerate(soup.p.next_siblings): print(i,child,end='\t') for i,child in enumerate(soup.p.previous_siblings): print(i,child,end='\t')
-
使用find\find_all方式:
- find( name , attrs , recursive , text , **kwargs )【根據(jù)參數(shù)來找出對(duì)應(yīng)的標(biāo)簽,但只返回第一個(gè)符合條件的結(jié)果】
-
find_all( name , attrs , recursive , text , **kwargs ):【根據(jù)參數(shù)來找出對(duì)應(yīng)的標(biāo)簽,但只返回所有符合條件的結(jié)果】
-
篩選條件參數(shù)介紹:
-
name:為標(biāo)簽名,根據(jù)標(biāo)簽名來篩選標(biāo)簽
-
attrs:為屬性,,根據(jù)屬性鍵值對(duì)來篩選標(biāo)簽,賦值方式可以為:屬性名=值,attrs={屬性名:值}【但由于class是python關(guān)鍵字,需要使用class_】
-
text:為文本內(nèi)容,根據(jù)指定文本內(nèi)容來篩選出標(biāo)簽,【單獨(dú)使用text作為篩選條件,只會(huì)返回text,所以一般與其他條件配合使用】
-
recursive:指定篩選是否遞歸,當(dāng)為False時(shí),不會(huì)在子結(jié)點(diǎn)的后代結(jié)點(diǎn)中查找,只會(huì)查找子結(jié)點(diǎn)
-
-
獲取到結(jié)點(diǎn)后的結(jié)果是一個(gè)bs4.element.Tag對(duì)象,所以對(duì)于獲取屬性、文本內(nèi)容、標(biāo)簽名等操作可以參考前面“使用標(biāo)簽篩選結(jié)果”時(shí)涉及的方法
from bs4 import BeautifulSoup html = """ <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <p class="news"><a >123456</a> <a id='i2'>78910</a> </p><p class="contents" id="i1"></p> <a rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" >advertisements</a> <span>aspan</span> </body> </html> """ soup = BeautifulSoup(html, 'lxml') print("---------------------") print(soup.find_all('a'),end='\n\n') print(soup.find_all('a')[0]) print(soup.find_all(attrs={'id':'i1'}),end='\n\n') print(soup.find_all(class_='news'),end='\n\n') print(soup.find_all('a',text='123456'))# print(soup.find_all(id='i2',recursive=False),end='\n\n')# a=soup.find_all('a') print(a[0].name) print(a[0].text) print(a[0].attrs)
-
使用select篩選【select使用CSS選擇規(guī)則】:
- soup.select(‘標(biāo)簽名'),代表根據(jù)標(biāo)簽來篩選出指定標(biāo)簽
- CSS中#xxx代表篩選id,soup.select(‘#xxx')代表根據(jù)id篩選出指定標(biāo)簽,返回值是一個(gè)列表
- CSS中.###代表篩選class,soup.select('.xxx')代表根據(jù)class篩選出指定標(biāo)簽,返回值是一個(gè)列表
- 嵌套select: soup.select(“#xxx .xxxx”),如(“#id2 .news”)就是id=”id2”標(biāo)簽下class=”news的標(biāo)簽,返回值是一個(gè)列表
- 獲取到結(jié)點(diǎn)后的結(jié)果是一個(gè)bs4.element.Tag對(duì)象,所以對(duì)于獲取屬性、文本內(nèi)容、標(biāo)簽名等操作可以參考前面“使用標(biāo)簽篩選結(jié)果”時(shí)涉及的方法
from bs4 import BeautifulSoup html = """ <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <p class="news"><a >123456</a> <a id='i2'>78910</a> </p><p class="contents" id="i1"></p> <a rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" >advertisements</a> <span class="span1" id='i4'>aspan</span> </body> </html> """ soup = BeautifulSoup(html, 'lxml') sp1=soup.select('span')#返回結(jié)果是一個(gè)列表,列表的元素是bs4元素標(biāo)簽對(duì)象 print(soup.select("#i2"),end='\n\n') print(soup.select(".news"),end='\n\n') print(soup.select(".news #i2"),end='\n\n') print(type(sp1),type(sp1[0])) print(sp1[0].name)#列表里面的元素才是bs4元素標(biāo)簽對(duì)象 print(sp1[0].attrs) print(sp1[0]['class'])
補(bǔ)充4:
對(duì)于代碼不齊全的情況下,可以使用soup.prettify()來自動(dòng)補(bǔ)全,一般情況下建議使用,以避免代碼不齊。
from bs4 import BeautifulSoup html = """ <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <p class="news"><a >123456</a> <a id='i2'>78910</a> </p><p class="contents" id="i1"></p> <a rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" >advertisements</a> <span class="span1" id='i4'>aspan </html> """ soup = BeautifulSoup(html, 'lxml') c=soup.prettify()#上述html字符串中末尾缺少</span> 和 </body> print(c)
如果想要獲得更詳細(xì)的介紹,可以參考官方文檔,令人高興的是,有了比較簡易的中文版:
https://www.crummy.com/software/BeautifulSoup/bs4/doc/index.zh.html
更多關(guān)于Python相關(guān)內(nèi)容可查看本站專題:《Python Socket編程技巧總結(jié)》、《Python正則表達(dá)式用法總結(jié)》、《Python數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Python函數(shù)使用技巧總結(jié)》、《Python字符串操作技巧匯總》、《Python入門與進(jìn)階經(jīng)典教程》及《Python文件與目錄操作技巧匯總》
希望本文所述對(duì)大家Python程序設(shè)計(jì)有所幫助。
- Python中BeautifulSoup通過查找Id獲取元素信息
- python爬蟲beautifulsoup解析html方法
- Selenium結(jié)合BeautifulSoup4編寫簡單的python爬蟲
- Python基于BeautifulSoup爬取京東商品信息
- 使用Python爬蟲庫BeautifulSoup遍歷文檔樹并對(duì)標(biāo)簽進(jìn)行操作詳解
- Python爬蟲庫BeautifulSoup獲取對(duì)象(標(biāo)簽)名,屬性,內(nèi)容,注釋
- Python下利用BeautifulSoup解析HTML的實(shí)現(xiàn)
- Python如何使用BeautifulSoup爬取網(wǎng)頁信息
- Python BeautifulSoup [解決方法] TypeError: list indices must be integers or slices, not str
- python中bs4.BeautifulSoup的基本用法
- python使用BeautifulSoup與正則表達(dá)式爬取時(shí)光網(wǎng)不同地區(qū)top100電影并對(duì)比
- python BeautifulSoup庫的安裝與使用
相關(guān)文章
用Python中的__slots__緩存資源以節(jié)省內(nèi)存開銷的方法
這篇文章主要介紹了用Python中的__slots__通過緩存資源的方式以節(jié)省內(nèi)存開銷的方法,且示例代碼非常簡單,需要的朋友可以參考下2015-04-04Python學(xué)習(xí)筆記之For循環(huán)用法詳解
這篇文章主要介紹了Python學(xué)習(xí)筆記之For循環(huán)用法,結(jié)合實(shí)例形式詳細(xì)分析了Python for循環(huán)的功能、原理、用法及相關(guān)操作注意事項(xiàng),需要的朋友可以參考下2019-08-08Python ckeditor富文本編輯器代碼實(shí)例解析
這篇文章主要介紹了Python ckeditor富文本編輯器代碼實(shí)例解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-06-06基于Python實(shí)現(xiàn)配置熱加載的方法詳解
所謂的配置熱加載,也就是說當(dāng)服務(wù)收到配置更新消息之后,我們不用重啟服務(wù)就可以使用最新的配置去執(zhí)行任務(wù)。本文將介紹如何用Python實(shí)現(xiàn)配置熱加載,需要的可以參考一下2022-07-07python學(xué)習(xí)筆記之調(diào)用eval函數(shù)出現(xiàn)invalid syntax錯(cuò)誤問題
python是一門多種用途的編程語言,時(shí)常扮演腳本語言的角色。一般來說,python可以定義為面向?qū)ο蟮哪_本語言,這個(gè)定義把面向?qū)ο蟮闹С趾兔嫦蚰_本語言的角色融合在一起。很多時(shí)候,人們常常喜歡用“腳本”和不是語言來描述python的代碼文件。2015-10-10一步步教你用Python實(shí)現(xiàn)2048小游戲
相信2048這個(gè)游戲?qū)Υ蠹襾碚f一定不陌生,下面這篇文章就主要給大家介紹了怎么用Python實(shí)現(xiàn)2048小游戲,文中通過注釋與示例代碼介紹的很詳細(xì),相信對(duì)大家的理解和學(xué)習(xí)具有一定的參考借鑒價(jià)值,有需要的朋友們一起來看看吧。2017-01-01python如何利用cv2.rectangle()繪制矩形框
cv2.rectangle這個(gè)函數(shù)的作用是在圖像上繪制一個(gè)簡單的矩形,下面這篇文章主要給大家介紹了關(guān)于python如何利用cv2.rectangle()繪制矩形框的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-12-12