成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Pyquery:一個靈活方便的 HTML 解析庫

開發(fā) 前端
總的來說,pyquery 還是相當方便的,相比 bs4 多了更多的靈活性,而且速度也更快一些。當然 pyquery 還有一些功能我們沒有說,比如追加節(jié)點等等,但這些不常用,所以不再贅述。因為我們只是解析 HTML,能基于選擇器獲取想要的標簽以及屬性就足夠了。

楔子

在工作中難免會遇到解析 HTML 的場景,比如將網(wǎng)頁下載下來之后,要解析出里面圖片的路徑、指定標簽里的文本等等。

而 pyquery 專門負責做這件事,它是仿照 jquery 設計的,用起來非常方便。并且 pyquery 底層基于 lxml,而 lxml 是使用 Cython 實現(xiàn)的,所以 pyquery 的速度也有保證。

from pyquery import PyQuery

html = """
<body>
    <p>
        古明地覺的編程教室
    </p>
</body>
"""
p = PyQuery(html)
print(type(p))
"""
<class 'pyquery.pyquery.PyQuery'>
"""
# 打印 PyQuery 對象會直接顯示 HTML 內(nèi)容
print(p)
"""
<body>
    <p>
        古明地覺的編程教室
    </p>
</body>
"""

我們在獲取 HTML 之后,直接傳遞 PyQuery 中,然后通過屬性選擇器即可獲取指定的內(nèi)容。

另外除了傳遞 HTML 文本之外,還可以傳遞一個 URL,或者 HTML 文件。

from pyquery import PyQuery

# 傳遞一個 url, 會自動調(diào)用 urlopen 下載內(nèi)容
p1 = PyQuery(url="https://www.baidu.com", encoding="utf-8")
# 傳遞一個 html 文件, 會自動打開并讀取
p2 = PyQuery(filename="1.html")

后兩種方式其實不是很常用,我們一般還是會搭配 requests 或者 httpx,下載完頁面之后直接丟給 PyQuery。

接下來我們看看如何篩選指定的標簽,多說一句,我個人非常喜歡這個庫,在解析 HTML 的時候首先想到的就是它。

CSS 選擇器

pyquery 是模仿 jquery 設計的,顯然它也是通過類似于 CSS 選擇器的方式進行篩選,下面介紹一些常用的選擇器。

from pyquery import PyQuery

html = """
<body>
    <div class="div_cls1 div_cls2">
        <p>S 老師不想你們?yōu)榱怂齼蓴【銈?lt;/p>
        <p class="p_cls1">高老師總能分享出好東西</p>
        <div class="div_cls3">
            <p id="six_six_six">
                我也退了,都怪我說了不該說的
            </p>
        </div>
    </div>

    <div>
        <a ></a>
        <p>
            <a ></a>
        </p>
    </div>

    <div class="div_cls1">
        <span>嘿嘿嘿</span>
    </div>
</body>
"""
p = PyQuery(html)

我們以上面這個 HTML 為例,來看看相關(guān)操作。

基于標簽進行選擇

# 選擇所有的 p 標簽
print(p("p"))
"""
<p>S 老師不想你們?yōu)榱怂齼蓴【銈?lt;/p>
        <p class="p_cls1">高老師總能分享出好東西</p>
        <p id="six_six_six">
                我也退了,都怪我說了不該說的
            </p>
        <p>
            <a />
        </p>
    
"""

會選擇所有指定的標簽,并且包含標簽里面的內(nèi)容。

同時選擇多個標簽

在基于標簽選擇時,也可以同時選擇多個標簽。

# 選擇所有的 p 標簽和 a 標簽
print(p("p,a"))
"""
<p>S 老師不想你們?yōu)榱怂齼蓴【銈?lt;/p>
        <p class="p_cls1">高老師總能分享出好東西</p>
        <p id="six_six_six">
                我也退了,都怪我說了不該說的
            </p>
        <a />
        <p>
            <a />
        </p>
    <a />
    
"""

多個標簽之間使用逗號分隔,會將多個標簽都篩選出來。

注意:篩選的標簽之間是獨立的,比如第二個 a 標簽,它在 p 標簽里面。我們篩選 p 標簽的時候,已經(jīng)將它內(nèi)部的 a 標簽篩選出來了,但在篩選 a 標簽的時候又篩選出來一次,因此標簽之間是獨立的。

選擇指定標簽下的子標簽

# 選擇所有的 div 標簽下的所有 a 標簽
print(p("div a"))
"""
<a />
        <a />
        
"""

多個標簽使用空格分隔,表示篩選層級,比如 tag1 tag2 tag3,表示篩選所有 tag1 標簽下的所有 tag2 標簽下的所有 tag3 標簽。

div a 表示從所有 div 的子孫節(jié)點中選擇 a 標簽,如果只希望從兒子節(jié)點中選擇呢?

# 選擇所有的 div 標簽下的所有 a 標簽,但只從兒子節(jié)點中選擇
# 第二個 a 標簽的外部套了個 p 標簽,所以不符合篩選條件
print(p("div>a"))
"""
<a />
        
"""

當標簽之間是空格,那么會從子孫節(jié)點當中選擇;當標簽之間是大于號,那么只會從兒子節(jié)點當中選擇。

按照 id 選擇標簽

# 選擇 id = "six_six_six" 的標簽
print(p("#six_six_six"))
"""
<p id="six_six_six">
                我也退了,都怪我說了不該說的
            </p>
"""

id 在一個 html 中具有唯一性,所以有 id 屬性的話,那么會非常好定位。

按照 class 選擇標簽

p = PyQuery(html)
# 選擇 class 等于 "p_cls1" 的標簽
print(p(".p_cls1"))
"""
<p class="p_cls1">高老師總能分享出好東西</p>
"""

選擇所有 class 屬性等于 p_cls1 的標簽,但是注意,一個標簽可以同時擁有多個 class。

print(p(".div_cls1"))
"""
<div class="div_cls1 div_cls2">
        <p>S 老師不想你們?yōu)榱怂齼蓴【銈?lt;/p>
        <p class="p_cls1">高老師總能分享出好東西</p>
        <div class="div_cls3">
            <p id="six_six_six">
                我也退了,都怪我說了不該說的
            </p>
        </div>
    </div>

    <div class="div_cls1">
        <span>嘿嘿嘿</span>
    </div>
        
"""

我們看到兩個 div 都應用了 div_cls1 這個 class,因此它們都被篩選了出來。而第一個 div 除了 div_cls1,還應用了 div_cls2 這個 class。

那么問題來了,如果我們希望選擇同時應用了 div_cls1 和 div_cls2 的標簽該怎么做呢?

print(p(".div_cls1.div_cls2"))
"""
<div class="div_cls1 div_cls2">
        <p>S 老師不想你們?yōu)榱怂齼蓴【銈?lt;/p>
        <p class="p_cls1">高老師總能分享出好東西</p>
        <div class="div_cls3">
            <p id="six_six_six">
                我也退了,都怪我說了不該說的
            </p>
        </div>
    </div>
        
"""

我們看到此時就只獲取了第一個 div,注意:.div_cls1 和 .div_cls2 之間不可以有空格,如果加上了空格,那么含義就變成了選擇 .div_cls1 標簽下面的 .div_cls2 標簽。

所以 id、class、標簽等選擇器,它們可以搭配使用。比如說:

圖片圖片

實際舉例說明:

# 找到所有 class 包含 div_cls1、div_cls2 的標簽
# 再從其兒子節(jié)點中找到所有 class 包含 .div_cls3 的 div 標簽
print(p("div.div_cls1.div_cls2>div.div_cls3"))
"""
<div class="div_cls3">
            <p id="six_six_six">
                我也退了,都怪我說了不該說的
            </p>
        </div>
        
"""

綜上所述,pyquery 還是很強大的。

選擇是否具有指定屬性的標簽

# 選擇具有 class 屬性的 p 標簽
print(p("p[class]"))
"""
<p class="p_cls1">高老師總能分享出好東西</p>
"""

# 選擇具有 id 屬性的 p 標簽
print(p("p[id]"))
"""
<p id="six_six_six">
                我也退了,都怪我說了不該說的
            </p>
"""

# 選擇 class="div_cls1" 的 div 標簽
# 等號右面可以是雙引號,也可以是單引號,也可以不加引號
print(p("div[class='div_cls1']"))
"""
<div class="div_cls1">
        <span>嘿嘿嘿</span>
    </div>
"""
# 注意:div[class='div_cls1'] 和 div.div_cls1 不同
# 前者要求 class 屬性必須為 div_cls1
# 而后者要求 class 屬性只要包含 div_cls1 即可


# 這些屬性除了 id、class 之外, 也可以是其它的任意屬性(隨便寫一個也可以)
# 下面選擇所有具有 href 屬性的 a 標簽
print(p("a[href]"))
"""
<a />
        <a />
"""

# 選擇 href 等于某個 url 的 a 標簽, 這里的 url 必須要使用引號包起來
print(p("a[))
"""
<a />
"""

# 還可以指定以 ... 開頭
print(p("a[href^='http://www.me.org/image']"))
"""
<a />
"""

# 指定以 ... 結(jié)尾
print(p("a[href$='2.png']"))
"""
<a />
"""

# 包含 ...
print(p("a[href*='bento']"))
"""
<a />
"""
# 當然其它屬性也可以,選擇 class 包含 div_cls1 的 a 標簽
# 此時 div[class*='div_cls1'] 和 div.div_cls1 是等價的
print(p("div[class*='div_cls1']") == p("div.div_cls1"))
"""
True
"""

選擇指定位置的標簽

# 先選擇所有 class 包含 div_cls1、div_cls2 的標簽
# 然后從它的兒子節(jié)點中選擇所有的 p 標簽
print(p(".div_cls1.div_cls2>p"))
"""
<p>S 老師不想你們?yōu)榱怂齼蓴【銈?lt;/p>
        <p class="p_cls1">高老師總能分享出好東西</p>
"""
# 先選擇所有 class 包含 div_cls1、div_cls2 的標簽
# 然后從它的兒子節(jié)點中選擇所有 class 等于 p_cls1 的 p 標簽
print(p(".div_cls1.div_cls2>p[class='p_cls1']"))
"""
<p class="p_cls1">高老師總能分享出好東西</p>
"""

# 然后也可以按照位置進行選擇,比如這里選擇符合條件的第一個 p 標簽
print(p(".div_cls1.div_cls2>p:nth-child(1)"))
"""
<p>S 老師不想你們?yōu)榱怂齼蓴【銈?lt;/p>
"""
# 選擇符合條件的第二個 p 標簽
print(p(".div_cls1.div_cls2>p:nth-child(2)"))
"""
<p class="p_cls1">高老師總能分享出好東西</p>
"""

選擇兄弟標簽

# 選擇 class 包含 p_cls1 的所有 p 標簽,然后選擇它的兄弟標簽
print(p("p.p_cls1").siblings())
"""
<p>S 老師不想你們?yōu)榱怂齼蓴【銈?lt;/p>
        <div class="div_cls3">
            <p id="six_six_six">
                我也退了,都怪我說了不該說的
            </p>
        </div>
"""

以上就是一些常見的 CSS 選擇器,在工作中一般是夠用了。

獲取標簽屬性

基于 CSS 選擇器,我們可以拿到指定的標簽,然后就是獲取屬性了,比如獲取文本。

print(p("p").text())
"""
S 老師不想你們?yōu)榱怂齼蓴【銈?高老師總能分享出好東西 我也退了,都怪我說了不該說的
"""

返回的是字符串,里面包含了所有的 p 標簽里的文本。但這樣我們就不知道,哪個文本是哪個 p 標簽里面的了,因此我們可以進行遍歷。

PyQuery 這個類繼承 list,因為基于選擇器篩選到的標簽可能會有多個,因此提供了用于遍歷的方法。但遍歷得到依舊是 PyQuery 對象,只不過此時里面就只有一個標簽了。

# 可以對選擇的標簽進行遍歷
for tag in p("p").items():
    print(tag.text())
"""
S 老師不想你們?yōu)榱怂齼蓴【銈?高老師總能分享出好東西
我也退了,都怪我說了不該說的
"""

text 方法用于獲取文本,至于其它屬性則通過 attr 方法獲取。

for tag in p("a").items():
    print(tag.attr("href"))
"""
http://www.me.org/bento/1.png
http://www.me.org/image/2.png
"""

for tag in p("div").items():
    print(tag.attr("class"))
"""
div_cls1 div_cls2
div_cls3
None
div_cls1
"""

# 遍歷所有的標簽,獲取 id 的值
for tag in p("*").items():
    if tag.attr("id") is not None:
        print(tag.attr("id"))
"""
six_six_six
"""

# 通過 attr 可以獲取所有的屬性,甚至自定義的也可以

是不是很方便呢?基于 CSS 選擇器和 attr 方法,我們就能獲取所有想要的屬性。

find 和 filter

PyQuery 對象還有兩個很重要的方法,分別是 find 和 filter。

先來看看 find:

# p("div .div_cls3 p") 等價于 p.find("div").find(".div_cls3").find("p")
# 或者也等價于 p.find("div").find(".div_cls3 p")
# 也等價于 p.find("div .div_cls3").find("p")
print(p("div .div_cls3 p") ==
      p.find("div").find(".div_cls3").find("p") ==
      p.find("div").find(".div_cls3 p") ==
      p.find("div .div_cls3").find("p"))
"""
True
"""

# 相信你應該明白 find 方法是做什么的了,它是基于指定條件繼續(xù)向內(nèi)篩選
# 比如我們成功篩選了指定的標簽
tag = p("div .div_cls3")
# 這時候想在 tag 的基礎上繼續(xù)獲取它內(nèi)部的 p 標簽,那么可以調(diào)用 find
print(tag.find("p"))
"""
<p id="six_six_six">
                我也退了,都怪我說了不該說的
            </p>
"""

tag.find 是在 tag 的基礎上繼續(xù)向內(nèi)篩選,而 tag.filter 則是對 tag 進行過濾。

tag = p("div p")
# 在 tag 的基礎上向內(nèi)篩選,獲取 class 包含 p_cls1 的標簽
# 但 div p 內(nèi)部沒有 class 包含 p_cls1 的標簽
print(tag.find(".p_cls1"))
"""
"""
# 對 tag 進行過濾,從已獲取的 tag 中過濾出 class 包含 p_cls1 的標簽
print(tag.filter(".p_cls1"))
"""
<p class="p_cls1">高老師總能分享出好東西</p>
"""

所以當你篩選了指定的 div 之后,你想從它的內(nèi)部繼續(xù)篩選,那么就使用 find 方法。如果你想按照指定條件對 div 進行過濾,那么就使用 filter。

另外 filter 還有一個用法,就是可以根據(jù)文本進行過濾。

print(p("p"))
"""
<p>S 老師不想你們?yōu)榱怂齼蓴【銈?lt;/p>
        <p class="p_cls1">高老師總能分享出好東西</p>
        <p id="six_six_six">
                我也退了,都怪我說了不該說的
            </p>
        <p>
            <a />
        </p>
"""
# 對篩選到 p 標簽進行過濾,只保留文本包含 "老師" 的 p 標簽
print(
    p("p").filter(lambda _, this: "老師" in PyQuery(this).text())
)
"""
<p>S 老師不想你們?yōu)榱怂齼蓴【銈?lt;/p>
        <p class="p_cls1">高老師總能分享出好東西</p>
"""

以上就是 find 和 filter 的用法,當你的解析需求不復雜時,直接調(diào)用 PyQuery 對象即可,否則可以搭配這兩個方法。

小結(jié)

總的來說,pyquery 還是相當方便的,相比 bs4 多了更多的靈活性,而且速度也更快一些。

當然 pyquery 還有一些功能我們沒有說,比如追加節(jié)點等等,但這些不常用,所以不再贅述。因為我們只是解析 HTML,能基于選擇器獲取想要的標簽以及屬性就足夠了。

雖然 pyquery 是仿照 jquery 設計的,但我們不會像 jquery 操作 DOM 那樣,對節(jié)點進行新增修改啥的。我們要做的只有查詢,基于選擇器獲取指定標簽,并且選擇器也不止我們上面介紹的那些,不過基本上夠用了。

責任編輯:武曉燕 來源: 古明地覺的編程教室
相關(guān)推薦

2021-10-14 09:14:50

PyQueryPython網(wǎng)頁

2023-08-01 07:25:38

Expresso框架API

2020-12-08 06:23:05

LockSupport線程工具

2025-04-29 10:28:25

2024-01-31 08:16:38

IPythonPython解釋器

2017-08-21 16:36:12

語法樹AST解析器HTML5

2023-11-03 11:57:04

2022-05-12 09:17:06

SQLPython

2009-04-28 13:25:36

Ajax函數(shù)Java

2012-09-24 11:11:32

HTML5游戲開發(fā)JavaScript

2024-12-19 08:58:50

2024-03-19 13:51:31

JavaScript插件

2022-09-30 13:57:15

JSON解析C語言

2020-04-22 18:17:01

樹莓派SD卡鏡像Linux

2024-02-19 08:26:59

wxPython界面庫開發(fā)

2025-06-03 10:00:00

LiteLLMPython

2025-06-04 08:05:00

Peewee?數(shù)據(jù)庫開發(fā)

2025-06-09 10:15:00

FastAPIPython

2025-06-05 08:10:00

PyneconePythonWeb 應用

2025-06-03 08:30:00

PotteryRedisPython
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 欧美中文字幕一区二区三区亚洲 | 在线亚洲免费视频 | 在线视频中文字幕 | 日本久久精品视频 | 欧美精品一区在线 | 欧美精品国产精品 | 久久精品男人的天堂 | 国产在线精品一区二区三区 | 久久久久久免费观看 | 精品欧美一区二区三区久久久小说 | 久久久久久成人 | 欧美午夜精品 | 久久尤物免费一区二区三区 | 人人人干 | 日韩高清一区 | 国产一区二区激情视频 | 99久久婷婷国产综合精品电影 | 激情网五月天 | 日韩免费一区二区 | 精产嫩模国品一二三区 | 成人欧美一区二区三区在线播放 | 亚洲免费人成在线视频观看 | 国产在线观看一区二区 | 欧美亚洲激情 | 亚洲国产精品久久久 | 国产成人综合亚洲欧美94在线 | 欧美一区二区三区 | 亚洲国产欧美在线 | 99热首页| 精品国产一区二区三区成人影院 | 欧美一级淫片免费视频黄 | 久久精品一区二区 | 国产成人精品一区二区三区四区 | 中文字幕视频在线观看免费 | www.久久 | 久久久看 | 在线观看毛片网站 | 国际精品鲁一鲁一区二区小说 | 午夜精品久久久久久 | 日本午夜精品 | 人人九九 |