• Class 14


    与Beautifu Soup 一样,初始化 pyquery 的时候,也需要传入 HTML 文本来初始化一个 PyQuery 对象。初始化方式有多种,如:直接传入字符串,传入 URL ,传人文件名等。

    1. 初始化
      • 字符串初始化  
        • html = '''
          <div>
          <ul>
          <li class="item-0">first item</li>
          <li class="item-1"><a href="link2.html">second item</a></li>
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></li>
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>
          <li class="item-0"><a href="link5.html">fifth item</a></li>
          </ul>
          </div>
          '''
          from pyquery import PyQuery as pq
          doc =pq(html)
          print(doc('li'))
          输出:
          <li class="item-0">first item</li>
          <li class="item-1"><a href="link2.html">second item</a></li>
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>
          <li class="item-0"><a href="link5.html">fifth item</a></li>
          View Code

          引PyQuery对象,取别名为 pq。声明了一个长 HTML字符串,并将其当作参数传递给 PyQuery 类,这样就成功完成了初始化。接下来,将初始化的对象传入 css 选择器。传入 li 节点,这样就可 选择所有的 li 节点。  

      • URL初始化
        • 初始化参数不仅可以以字符串的形式传递,可以传入网页的URL,此时只需要指定参数为 url 即可:
          from pyquery import PyQuery as pq
          doc =pq(url='https://cuiqingcai.com')
          print(doc('title'))
          输出:
          <title>静觅丨崔庆才的个人博客</title>&#13;
          View Code

          PyQuery 对象会首先请求这个 URL,用得到的 HTML内容完成初始化,相当于用网页的源代码以字符串的形式传递给PyQuery 类来初始化。与以下功能相同:

          from pyquery import PyQuery as pq
          import requests
          doc =pq(requests.get(url='https://cuiqingcai.com').text)
          print(doc('title'))
      • 文件初始化

        • 除了传递 URL,还可以传递本地的文件名,此时将参数指定为filename即可:

          from pyquery import PyQuery as pq
          doc =pq(filename='demo.html')
          print(doc('li'))

          这里需要有一个本地 HTML 文件 demo.html,其内容是待解析 HTML 字符串。它会首先读取本地的文件内容,再用文件内容以字符串形式传递给PyQuery 类来初始化。

    2. 基本CSS选择器
      • pyquery的CSS选择示例:
        from pyquery import PyQuery as pq
        doc =pq('html')
        print(doc('#container .list li'))
        print(type(doc('#container .list li')))
        输出:
        <li class="item-0">first item</li>
        <li class="item-1"><a href="link2.html">second item</a></li>
        <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
        <li class="item-1 active"><a href="link4.html">fourth item</a></li>
        <li class="item-0"><a href="link5.html">fifth item</a></li>
        
        <class 'pyquery.pyquery.PyQuery'>
        View Code

        初始化 PyQuery 对象后,传入了 CSS 选择器#container .list li ,意思是先选取 id container 的节点,然后再选取其内部的 class 为list 的节点内部的所有 li 节点。然后输出,成功获取到了符合条件的节点。 它的类型依然是 PyQuery 类型。

    3. 查找节点

      • 子节点

        • 查找子节点时,需要用到干find()方法,此时传人的参数是 CSS 选择器。前面的 HTML 为例:

          from pyquery import PyQuery as pq
          doc =pq(html)
          items =doc('.list')
          print(type(items))
          print(items)
          lis = items.find('li')
          print(type(lis))
          print(lis)
          输出:
          <class 'pyquery.pyquery.PyQuery'>
          <ul class="list">
          <li class="item-0">first item</li>
          <li class="item-1"><a href="link2.html">second item</a></li>
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>
          <li class="item-0"><a href="link5.html">fifth item</a></li>
          </ul>
          
          <class 'pyquery.pyquery.PyQuery'>
          <li class="item-0">first item</li>
          <li class="item-1"><a href="link2.html">second item</a></li>
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>
          <li class="item-0"><a href="link5.html">fifth item</a></li>
          View Code

          首先,选取 class 为 list 的节点,再调用了 find()方法,传入 CSS 选择器,选取其内部 li 节点,最后输出。find()方法会将符合条件的所有节点选择出来,结果是 PyQuery 类型。

          其实find()的查找范围是节点的所有子孙节点,如果只想查找子节点,那么可以用 children()方法:

          lis = items.children()
          print(type(lis))
          print(lis)
          输出:
          <class 'pyquery.pyquery.PyQuery'>
          <li class="item-0">first item</li>
          <li class="item-1"><a href="link2.html">second item</a></li>
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>
          <li class="item-0"><a href="link5.html">fifth item</a></li>
          View Code

          如果要筛选所有子节点中符合条件的节点,如:筛选出子节点中 class 为 active 节点,可以向 children()方法传入 CSS 选择器 .active:

          lis = items.children('.active')
          print(lis)
          输出:
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>
          View Code

          输出结果已经做了筛选,留下了 class为 active 的节点。

      • 父节点

        • 可以用 parent()方法来获取某个节点的父节点,示例:

          html = '''
          <div class="wrap">
          <div id="container">
          <ul class="list">
          <li class="item-0">first item</li>
          <li class="item-1"><a href="link2.html">second item</a></li>
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>
          <li class="item-0"><a href="link5.html">fifth item</a></li>
          </ul>
          </div>
          </div>
          '''
          from pyquery import PyQuery as pq
          doc=pq(html)
          items =doc('.list')
          container =items.parent()
          print(type(container))
          print(container)
          输出:
          <class 'pyquery.pyquery.PyQuery'>
          <div id="container">
          <ul class="list">
          <li class="item-0">first item</li>
          <li class="item-1"><a href="link2.html">second item</a></li>
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>
          <li class="item-0"><a href="link5.html">fifth item</a></li>
          </ul>
          </div>
          View Code

          首先用 .list 选取 class为 list 的节点,然后调用 parent()方法得到其父节点,其类型依然是 PyQuery 类型。

          父节点是该节点的直接父节点,也就是说,它不会再去查找父节点的父节点, 即祖先节点  

          如果想获取某个祖先节点,这时可以用 parents()方法:

          from pyquery import PyQuery as pq
          doc=pq(html)
          items=doc('.list')
          parents =items.parents()
          print(parents)
          输出:
          <div class="wrap">
          <div id="container">
          <ul class="list">
          <li class="item-0">first item</li>
          <li class="item-1"><a href="link2.html">second item</a></li>
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>
          <li class="item-0"><a href="link5.html">fifth item</a></li>
          </ul>
          </div>
          </div><div id="container">
          <ul class="list">
          <li class="item-0">first item</li>
          <li class="item-1"><a href="link2.html">second item</a></li>
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>
          <li class="item-0"><a href="link5.html">fifth item</a></li>
          </ul>
          </div>
          View Code

          输出结果有两个:一个是 class为wrap 的节点,一个是 id 为 container 的节点。parents()方法会返回所有的祖先节点。

          如果想要筛选某个祖先节点,可以向 parents()方法传入 CSS 选择器,这样就会返回祖先节 点中符合 CSS 选择器的节点:

          from pyquery import PyQuery as pq
          doc=pq(html)
          items=doc('.list')
          parent=items.parents('.wrap')
          print(parent)
          输出:
          <div class="wrap">
          <div id="container">
          <ul class="list">
          <li class="item-0">first item</li>
          <li class="item-1"><a href="link2.html">second item</a></li>
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>
          <li class="item-0"><a href="link5.html">fifth item</a></li>
          </ul>
          </div>
          </div>
          View Code

          输出结果只保留了 class 为 wrap 的节点

      • 兄弟节点

        • 如果要获取兄弟节点,可以使用 siblings()方法。以上面 TML 为例:

          from pyquery import PyQuery as pq
          doc =pq(html)
          li =doc('.list .item-0.active')
          print(li.siblings())
          输出:
          <li class="item-1"><a href="link2.html">second item</a></li>
          <li class="item-0">first item</li>
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>
          <li class="item-0"><a href="link5.html">fifth item</a></li>
          View Code

          首先选择 class为 list 的节点内部 class 为 item-0 和 active 的节点,也就是第三个 li 节点。它的兄弟节点有 4 个,那就是第一、二、四、五个 li 节点。

          如果要筛选某个兄弟节点,依然可以向 siblings 方法传入 css 选择器,这样就会从所有兄弟节点中挑选符合条件的节点了:

          from pyquery import PyQuery as pq
          doc =pq(html)
          li =doc('.list .item-0.active')
          print(li.siblings(' .active'))
          输出:
          <li class="item-1 active"><a href="link4.html">fourth item</a></li>

          筛选了 class 为 active 的节点。

    4. 遍历    

      • pyquery 选择结果可能是多个节点,也可能是单个节点。都是PyQuery 类型,并没有返回像 Beautiful Soup 那样的列表。  

        对于单个节点来说,可以直接打印输出,也可以直接转成字符串

        from pyquery import PyQuery as pq
        doc = pq(html)
        li = doc('.item-0.active')
        print(li)
        print(str(li))
        输出:
        <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
        
        <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
        View Code

        对于多个节点的结果,就需要遍历来获取。如,这里把每一个 li 节点进行遍历,需要调用 items()方法:

        from pyquery import PyQuery as pq
        doc= pq(html)
        lis = doc('li').items()
        for li in lis:
            print(li,type(li))
        输出:
        <li class="item-0">first item</li>
         <class 'pyquery.pyquery.PyQuery'>
        <li class="item-1"><a href="link2.html">second item</a></li>
         <class 'pyquery.pyquery.PyQuery'>
        <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
         <class 'pyquery.pyquery.PyQuery'>
        <li class="item-1 active"><a href="link4.html">fourth item</a></li>
         <class 'pyquery.pyquery.PyQuery'>
        <li class="item-0"><a href="link5.html">fifth item</a></li>
         <class 'pyquery.pyquery.PyQuery'>
        View Code

        调用 items()方法后, 会得到生成器, 遍历, 可以逐个得到 li 节点,也是PyQuery 类型。每个 li 节点还可以调用前面所说的方法进行选择,如:继续查询子节点,寻找某个祖先节点等。

    5. 获取信息  

      • 提取节后,最终目是提取节点所包含的信息。比较重要的信息有两类,一是获取属’性,二是获取文本

      • 获取属性
        • 提取到某个 PyQuery 类型的节点后,就可以调用 attr ()方法来获取属性:
          from pyquery import PyQuery as pq
          doc =pq(html)
          a=doc('.item-0.active a')
          print(a,type(a))
          print(a.attr('href'))
          输出:
          <a href="link3.html"><span class="bold">third item</span></a> <class 'pyquery.pyquery.PyQuery'>
          link3.html
          View Code

          首先选中 class 为 item-0 和 active 的 li 节点内的 a 节点,是 PyQuery 类型。然后调用 attr()方法。在这个方法中传入属性的名称,就可以得到这个属性值了。

          也可以通过调用 attr 属性来获取属性,用法:
          print(a.attr.href)
          输出:
          link3.html

          这两种方法的结果完全一样。

          如果选中的是多个元素,然后调用 attr()方法:
          from pyquery import PyQuery as pq
          doc = pq(html)
          a = doc('a')
          print(a,type(a))
          print(a.attr('href'))
          print(a.attr.href)
          输出:
          <a href="link2.html">second item</a><a href="link3.html"><span class="bold">third item</span></a><a href="link4.html">fourth item</a><a href="link5.html">fifth item</a> <class 'pyquery.pyquery.PyQuery'>
          link2.html
          link2.html
          View Code

          选中的 a 节点应该有 4个,而且打印结果也应该是 4 个,但是当我们调用 attr() 方法时,返回结果却只是第一个。因为,当返回结果包含多个节点时,调用attr()方法,只会得到第一个节点的属性

          遇到这种情况,如果想获取所有的 a 节点的属性,就要用到前面的遍历:
          from pyquery import PyQuery as pq
          doc =pq(html)
          a= doc('a')
          for item in a.items():
              print(item.attr('href'))
          输出:
          link2.html
          link3.html
          link4.html
          link5.html
          View Code

          在进行属性获取时,可以观察返回节点是一个还是多个,如果是多个, 需要遍历才能依次获取每个节点的属性。

      • 获取文本
        • 获取节点后的另一个主要操作就是获取其内部的文本,此时可以调用 text()方法来实现:
          from pyquery import PyQuery as pq
          doc =pq(html)
          a =doc('.item-0.active a')
          print(a)
          print(a.text()
          输出:
          <a href="link3.html"><span class="bold">third item</span></a>
          third item
          View Code

          这里首先选中一个 a 节点,然后调用 text()方法,就可以获取其内部的文本信息。此时它会忽略掉节点内部包含的所有 HTML ,只返回纯文字内容

          如果想要获取这个节点内部的 HTML 文本,就要用 html()方法了:
          from pyquery import PyQuery as pq
          doc =pq(html)
          li =doc('.item-0.active')
          print(li)
          print(li.html())
          输出:
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          
          <a href="link3.html"><span class="bold">third item</span></a>
          View Code

          这里我们选中了第三个 li 节点,然后调用了 html()方法,返回的结果应该是 li 节点内的所有 HTML 文本。

          如果选中的结果是多个节点, text()或 html()返回结果各不相同,实例:
          from pyquery import PyQuery as pq
          doc = pq(html)
          li =doc('li')
          print(li.html())
          print(li.text())
          print(type(li.text()))
          输出:
          first item
          first item second item third item fourth item fifth item
          <class 'str'>
          View Code

          html()方法返回的是第一个 li 节点的内部 HTML 文本,而 text()返回所有 li 节点内部的纯文本,中间用 个空格分割开,即返回结果是一个字符串。

          注意:如果得到的结果是多个节点,且想要获取每个节点的内部 HTML 文本, 则需要遍历每个节点。text ()方法不需要遍历就可以获取,它将所有节点取文本之后合并成 一个字符串。
    6. 节点操作
      • pyquery 提供了一系列方法来对节点进行动态修改,如为某个节点添加一个 class,移除某个节点等,例:
      • addClass()和 removeClass()方法可以动态改变节点的 class 属性
        • from pyquery import PyQuery as pq
          doc = pq(html)
          li = doc('.item-0.active')
          print(li)
          li.removeClass('active')
          print(li)
          li.addClass('active')
          print(li)
          输出:
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          
          <li class="item-0"><a href="link3.html"><span class="bold">third item</span></a></li>
          
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          View Code

          首先选中了第三个 li 节点,然后调用 removeClass()方法,将 li 节点的 active 这个 class 移除,后来在调用 addClass()方法,将 class 添加回来。每执行一次操作,就打印输出当前 li 节点的内容

      • attr()、text() 和 html() 方法对属性进行操作,来改变节点内部的内容。 示例: 

        • 如果 attr()方法只传入第一个参数的属性名,则是获取这个属性值; 如果传入第二个参 数,可以用来修改属性值。 text()和 html()方法如果不传参数,则是获取节点内纯文本和 HTML 文本; 如果传人参数,则进行赋值。

        • html = '''
          <ul class="list">
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          </ul>
          '''
          from pyquery import PyQuery as pq
          doc = pq(html)
          li =doc('.item-0.active')
          print(li)
          li.attr('name','link')
          print(li)
          li.text('changed item')
          print(li)
          li.html('<span>changed item</span>')
          print(li)
          输出:
          <li class="item-0 active"><a href="link3.html"><span class="bold">third item</span></a></li>
          
          <li class="item-0 active" name="link"><a href="link3.html"><span class="bold">third item</span></a></li>
          
          <li class="item-0 active" name="link">changed item</li>
          
          <li class="item-0 active" name="link"><span>changed item</span></li>
          View Code

          首先选中 li 节点,然后调用 attr()方法来修改属性,该方法的第一个参数为属性名,第二个参数为属性值。 接着,调用 text()和 html()方法来改变节点内部的内容。三次操作后,分 别打印输出当前的 li 节点。

          调用 attr()方法后,li 节点多了一个原本不存在的属性 name, 其值为 link。 接着调 用 text()方法,传人文本之后, li 节点内部的文本全被改为传人的字符串文本了。 最后,调用 html() 方法传人 HTML 文本后 li 节点内部又变为传人的 HTML 文本了。

      • remove()
        • html = '''
          <div class="wrap">
              Hello,World
          <p>This is a paragraph.</p>
          </div>
          '''
          from pyquery import PyQuery as  pq
          doc =pq(html)
          wrap = doc('.wrap')
          print(wrap.text())
          输出:
          Hello,World
          This is a paragraph.
          View Code

          因为text()把所有的存文本都提取出来,如果需要提取Hello,world这个字符串,而不需要P节点内部的字符串,可使用remove.

          from pyquery import PyQuery as  pq
          doc =pq(html)
          wrap = doc('.wrap')
          wrap.find('p').remove()
          print(wrap.text())
          输出:
          Hello,World

          首先选中 p 节点,然后调用了 remove()方法将其移除,这时 wrap 内部就只剩下 Hello, World 这句话了,再利用 text()方法提取即可。 

          实还有很多节点操作的方法,比如 append()、 empty()和 prepend()等方法, jQuery 的用法完全一致,详细的用法可以参考官方文档: http://pyquery.readthedocs.io/en/latest/api.html
    7. 伪类选择器
      • CSS 选择器之所以强大,就是它支持多种多样的伪类选择器,如:选择第一个节点、最后一个节点、奇偶数节点、包含某一文本的节点等。 示例:
        from pyquery import PyQuery as pq
        doc = pq(html)
        li=doc('li:first-child')
        print(li)
        li =doc('li:last-child')
        print(li)
        li=doc('li:nth-child(2)')
        print(li)
        li=doc('li:gt(2)')
        print(li)
        li=doc('li:nth-child(2n)')
        print(li)
        li=doc('li:contains(second)')
        print(li)
        输出:
        <li class="item-0">first item</li>
        
        <li class="item-0"><a href="link5.html">fifth item</a></li>
        
        <li class="item-1"><a href="link2.html">second item</a></li>
        
        <li class="item-1 active"><a href="link4.html">fourth item</a></li>
        <li class="item-0"><a href="link5.html">fifth item</a></li>
        
        <li class="item-1"><a href="link2.html">second item</a></li>
        <li class="item-1 active"><a href="link4.html">fourth item</a></li>
        
        <li class="item-1"><a href="link2.html">second item</a></li>
        View Code

        这里使用 CSS3 的伪类选择器,依次选择了第一个 li 节点、最后一个 li 节点 、第二个 li 节点、第三个 li 之后的 li 节点、 偶数位置的 li 节点、 包含 second 文本的 li 节点。
        关于 css 选择器的更多用法,可以参考 http://www.w3school.com.cn/css/index.asp。
        更多内容,可以参考 pyquery 的官方文 档: http://pyquery.readthedocs.io

                  
  • 相关阅读:
    STL中string的源码解读
    Sublime插件:Terminal
    sublime text3安装Package Control
    [转]Sublime Text操作
    python itertools模块实现排列组合
    pandas 选择某几列
    更改pandas dataframe 列的顺序
    pandas之groupby分组与pivot_table透视表
    IPython notebook快捷键(Jupyter notebook)
    人生的意义
  • 原文地址:https://www.cnblogs.com/Mack-Yang/p/10098724.html
Copyright © 2020-2023  润新知