如何提取和忽略标记中的跨度? - Python

Posted

技术标签:

【中文标题】如何提取和忽略标记中的跨度? - Python【英文标题】:How to extract and ignore span in markup? - python 【发布时间】:2015-12-19 15:11:12 【问题描述】:

如何在 HTML 标记中提取和忽略 span?

我的输入如下所示:

<ul class="definitions">
<li><span>noun</span> the joining together of businesses which deal with different stages in the production or <a href="sale.html">sale</a> of the same <u slug="product">product</u>, as when a restaurant <a href="chain.html">chain</a> takes over a <a href="wine.html">wine</a> importer</li></ul>

期望的输出:

label = 'noun' # String embedded between <span>...</span>
meaning = 'the joining together of businesses which deal with different stages in the production or sale of the same product, as when a restaurant chain takes over a wine importer' # the text without the string embedded within <span>...</span>
related_to = ['sale', 'chain', 'wine'] # String embedded between <a>...</a>
utag = ['product'] # String embedded between <u>...</u>

我试过了:

>>> from bs4 import BeautifulSoup
>>> text = '''<ul class="definitions">
...     <li><span>noun</span> the joining together of businesses which deal with different stages in the production or <a href="sale.html">sale</a> of the same <u slug="product">product</u>, as when a restaurant <a href="chain.html">chain</a> takes over a <a href="wine.html">wine</a> importer</li></ul>'''
>>> bsoup = BeautifulSoup(text)
>>> bsoup.text
u'\nnoun the joining together of businesses which deal with different stages in the production or sale of the same product, as when a restaurant chain takes over a wine importer'

# Getting the `label`
>>> label = bsoup.find('span')
>>> label
<span>noun</span>
>>> label = bsoup.find('span').text
>>> label
u'noun'

# Getting the text.
>>> bsoup.text.strip()
u'noun the joining together of businesses which deal with different stages in the production or sale of the same product, as when a restaurant chain takes over a wine importer'
>>> bsoup.text.strip
>>> definition = bsoup.text.strip() 
>>> definition = definition.partition(' ')[2] if definition.split()[0] == label else definition
>>> definition
u'the joining together of businesses which deal with different stages in the production or sale of the same product, as when a restaurant chain takes over a wine importer'

# Getting the related_to and utag
>>> related_to = [r.text for r in bsoup.find_all('a')]
>>> related_to
[u'sale', u'chain', u'wine']
>>> related_to = [r.text for r in bsoup.find_all('u')]
>>> related_to = [r.text for r in bsoup.find_all('a')]
>>> utag = [r.text for r in bsoup.find_all('u')]
>>> related_to
[u'sale', u'chain', u'wine']
>>> utag
[u'product']

使用 BeautifulSoup 没问题,但获取所需内容有点冗长。

还有其他方法可以实现相同的输出吗?

是否有一些组的正则表达式来捕获所需的输出?

【问题讨论】:

【参考方案1】:

PyQuery 是使用 BeautifulSoup 的另一种选择。它遵循类似 jQuery 的语法,用于从 html 中提取信息。

另外,对于正则表达式...可以使用类似下面的内容。

import re

text = """<ul class="definitions"><li><span>noun</span> the joining together of businesses which deal with different stages in the production or <a href="sale.html">sale</a> of the same <u slug="product">product</u>, as when a restaurant <a href="chain.html">chain</a> takes over a <a href="wine.html">wine</a> importer</li></ul>"""

match_pattern = re.compile(r"""
                (?P<label>(?<=<span>)\w+?(?=</span>)) # create the label \
                                                         item for groupdict()
                 """, re.VERBOSE)

match = match_pattern.search(text)
match.groupdict()

输出:

'label': 'noun'

使用以上作为模板,您也可以在此基础上构建其他 html 标签。它使用(?P&lt;name&gt;...) 命名匹配的模式(即标签),然后使用(?=...) lookahead assertionpositive lookbehind assertion 来执行匹配。

此外,如果您有一个文档包含不止一次您提到的文本模式实例,请查看 findall 或 finditer。

【讨论】:

【参考方案2】:

它仍然有一个漂亮的格式良好的结构,并且你已经清楚地说明了规则集。我仍然会使用 BeautifulSoup 应用“提取方法”重构方法来处理它:

from pprint import pprint
from bs4 import BeautifulSoup


data = """
<ul class="definitions">
<li><span>noun</span> the joining together of businesses which deal with different stages in the production or <a href="sale.html">sale</a> of the same <u slug="product">product</u>, as when a restaurant <a href="chain.html">chain</a> takes over a <a href="wine.html">wine</a> importer</li></ul>
"""

def get_info(elm):
    label = elm.find("span")
    return 
        "label": label.text,
        "meaning": "".join(getattr(sibling, "text", sibling) for sibling in label.next_siblings).strip(),
        "related_to": [a.text for a in elm.find_all("a")],
        "utag": [u.text for u in elm.find_all("u")]
    

soup = BeautifulSoup(data, "html.parser")
pprint(get_info(soup.li))

打印:

'label': u'noun',
 'meaning': u'the joining together of businesses which deal with different stages in the production or sale of the same product, as when a restaurant chain takes over a wine importer',
 'related_to': [u'sale', u'chain', u'wine'],
 'utag': [u'product']

【讨论】:

以上是关于如何提取和忽略标记中的跨度? - Python的主要内容,如果未能解决你的问题,请参考以下文章

如何从 Python 3 中的字节数组中的特定位中提取值?

循环仅在添加跨度标记时显示数组中的最后一个字符串

如何从 python 中的图像(或 pdf 文件)中提取名称和手写数字?

如何通过python忽略正则表达式中的html注释标签

从 html 中的 <script> 标记中提取 var

遍历 HTML div 并使用 JavaScript 或 jQuery 从子跨度中提取值