21

我怎样才能得到第一个孩子?

 <div class="cities"> 
       <div id="3232"> London </div>
       <div id="131"> York </div>
  </div>

我怎样才能到达伦敦?

for div in nsoup.find_all(class_='cities'):
    print (div.children.contents)

AttributeError:“listiterator”对象没有属性“内容”

4

3 回答 3

10

div.children 返回一个迭代器。

for div in nsoup.find_all(class_='cities'):
    for childdiv in div.find_all('div'):
        print (childdiv.string) #london, york

'\n'AttributeError 被提出,因为像are in这样的非标签.children。只需使用适当的子选择器来查找特定的 div。

(更多编辑)无法重现您的异常 - 这是我所做的:

In [137]: print foo.prettify()
<div class="cities">
 <div id="3232">
  London
 </div>
 <div id="131">
  York
 </div>
</div>

In [138]: for div in foo.find_all(class_ = 'cities'):
   .....:     for childdiv in div.find_all('div'):
   .....:         print childdiv.string
   .....: 
 London 
 York 

In [139]: for div in foo.find_all(class_ = 'cities'):
   .....:     for childdiv in div.find_all('div'):
   .....:         print childdiv.string, childdiv['id']
   .....: 
 London  3232
 York  131
于 2013-03-19T03:02:53.997 回答
9

使用现代版本的 bs4(当然是 bs4 4.7.1+),您可以访问 :first-child css 伪选择器。尼斯和描述性的。如果soup.select_one您只想返回第一个匹配项,即使用soup.select_one('.cities div:first-child').text. not None在使用访问器之前进行测试被认为是一种很好的做法.text

from bs4 import BeautifulSoup as bs

html = '''
<div class="cities"> 
       <div id="3232"> London </div>
       <div id="131"> York </div>
  </div>
  '''
soup = bs(html, 'lxml') #or 'html.parser'
first_children = [i.text for i in soup.select('.cities div:first-child')]
print(first_children)
于 2019-09-06T05:14:47.790 回答
6

当前接受的答案得到所有城市,当问题只想要第一个时。

如果您只需要第一个孩子,您可以利用.children返回迭代器而不是列表的优势。请记住,迭代器会即时生成列表项,并且因为我们只需要迭代器的第一个元素,所以我们不需要生成所有其他城市元素(从而节省时间)。

for div in nsoup.find_all(class_='cities'):
    first_child = next(div.children, None)
    if first_child is not None:
        print(first_child.string.strip())
于 2019-09-06T02:41:40.510 回答