Beautiful Soup的使用

  • 获取页面: urllib, requests
  • 解析页面信息: 正则表达式, BeautifulSoup4(BS4)
  • BS4简介
    Beautiful Soup提供一些简单的、python式的函数用来处理导航、搜索、修改分析树等功能。它是一个
    工具箱,通过解析文档为tiful Soup自动将输入文档转换为Unicode编码,输出文档转换为utf-8编码。
    你不需要考虑编码方式,除非文档没有指定一个编一下原始编码方式就可以了。
1. 安装

pip bs4或pip beautifulsoup4

2. 使用
  1. 创建Beautiful Soup 对象

from bs4 import BeautifulSoup
soup=BeautifulSoup(str,‘lxml’)//str在下面的测试代码中

  1. 四大对象种类
    Beautiful Soup 将复杂HTML 文档转换成一个复杂的树形结构,每个节点都是Python对象,所有对象可以归纳为4种
  • Tag
  • NavigatableString
  • BeautifulSoup
  • Comment

2.1 根据标签获取内容
Tag,就是HTML中的一个标签,比如 < div>,< title>等,
注意:相同的标签只能获取第一个符合要求的标签

  • 获取标签

soup=BeautifulSoup(str,‘lxml’)
print(soup.title)
print(soup.title.name) # 标签的名称

  • 获取标签属性

print(soup.div.attrs)
print(soup.div.attrs[‘href’])
print(soup.a[‘href’])

  • 标签常用的方法

get方法用于得到标签下的属性值,注意这是一个重要的方法,在许多场合都能用到,比如你要得到img标签下的图像url,那么就可以用soup.img.get(‘src’)
print(soup.div.get(‘class’))
print(soup.div[‘class’])

string得到标签下的文本内容,只有在此标签下没有子标签,或者只有一个子标签的情况下才能返回其中的内容,否则返回的是None;
print(soup.a.string) # 标签里面的内容

get_text()可以获得一个标签中的所有文本内容,包括子孙节点的内容,这是最常用的方法
print(soup.a.get_text())

  • 对获取的属性信息进行修改

print(soup.a.get(‘href’))
soup.a[‘href’] = ‘http://www.baidu.com
print(soup.a.get(‘href’))
print(soup.a)

NavigatableString 获取内容

print(soup.strong.string)
print(soup.strong.text)

BeautifulSoup

BeautifulSoup 对象表示的是一个文档的全部内容。大部分时候,可以把它当作Tag对象,支持遍历,索索等方法

Comment

if(type(soup.strong.string)==Comment):
//print(soup.strong.text)
print(soup.strong.prettify())
else:
print(type(soup.strong.string))

2.2 面向对象的匹配
过滤器find_all()

查找符合条件的所有标签;
aTagObj = soup.find_all(‘a’)
print(aTagObj)
for item in aTagObj:
print(item)

需求: 获取所有的a标签, 并且类名为"sister"
aTagObj = soup.find_all(‘a’, class_=“sister”)

2.3. 根据内容进行匹配

print(soup.find_all(text=“story”))
print(soup.find_all(text=re.compile(‘story\d+’)))

补充

需要安装第三方模块lxml

  1. 返回符合条件的第一个标签内容
    print(soup.title)
    print(soup.p)
    print(soup.find(‘p’, class_=re.compile(r’^ti.*?’)))

  2. 返回符合条件的所有标签内容
    print(soup.find_all(‘p’))
    print(soup.find_all(‘p’, class_=‘title’, text=re.compile(r’.?story.?’)))

  3. 获取符合条件的p标签或者a标签
    print(soup.find([‘title’, ‘a’]))
    print(soup.find_all([‘title’, ‘a’]))
    print(soup.find_all([‘title’, ‘a’], class_=[‘title’, ‘sister’]))

  4. CSS匹配
    标签选择器
    print(soup.select(“title”))
    类选择器(.类名)
    print(soup.select(".sister"))
    id选择器(#id名称)
    print(soup.select("#link1"))
    #此处不支持正则表达式;
    #print(soup.select(re.compile("#link\d+")))
    属性选择器()
    print(soup.select(“input[type=‘password’]”))

bs4模块的解析器
官方中文文档: https://beautifulsoup.readthedocs.io/zh_CN/v4.4.0/

下表列出了主要的解析器,以及它们的优缺点:

解析器:使用方法:优势:劣势

  • Python标准库
    BeautifulSoup(markup, “html.parser”)
    Python的内置标准库
    执行速度适中
    文档容错能力强
    Python 2.7.3 or 3.2.2)前 的版本中文档容错能力差

  • lxml
    HTML 解析器 BeautifulSoup(markup, “lxml”)
    速度快
    文档容错能力强
    需要安装C语言库

  • lxml
    XML 解析器
    BeautifulSoup(markup, [“lxml-xml”])
    BeautifulSoup(markup, “xml”)
    速度快
    唯一支持XML的解析器
    需要安装C语言库

  • html5lib
    BeautifulSoup(markup, “html5lib”)
    最好的容错性
    以浏览器的方式解析文档
    生成HTML5格式的文档
    速度慢
    不依赖外部扩展

  1. 代码
from bs4 import BeautifulSoup
from bs4.element import Comment
str='''//这里举例用的,一般使用bs都是用爬来的代码,无需我们自己写
<title id="title">西邮</title>
<div class="info" float="left">welcome to xupt</div>
<div class="info" float="right">
   <span>Good Study</span>
   <a href="www.baidu.com"></a>
   <strong><!--注释!--></strong>
</div>
'''
soup=BeautifulSoup(str,'lxml')
printf('--------------测试--------------')//下面注释就是的输出结果
print(soup.title)	//<title id="title">西邮</title>
print(soup.div)	//<div class="info" float="left">welcome to xupt</div>
print(soup.div.attrs)/	/{'class': ['info'], 'float': 'left'}
print(soup.div.get('class'))	//['info']
print(soup.div['class'])	//['info']
print(soup.div.text)	//welcome to xupt
print(soup.div.string)	//welcome to xupt

print(soup.a['href'])	//www.baidu.com

print(soup.strong.string)	//注释!
print(type(soup.strong.string))		//<class 'bs4.element.Comment'>

if(type(soup.strong.string)==Comment):
   //print(soup.strong.text)
   print(soup.strong.prettify())
else:
   print(type(soup.strong.string))
 /*结果
 <strong>
<!--注释!-->
</strong>
*/
print('------------------find_all------------------')
print(soup.find_all('title'))	//[<title id="title">西邮</title>]
print(soup.find_all(id='title'))	//[<title id="title">西邮</title>]
print(soup.find_all(class_='info'))	
/*
[<div class="info" float="left">welcome to xupt</div>, <div class="info" float="right">
<span>Good Study</span>
<a href="www.baidu.com"></a>
<strong><!--注释!--></strong>
</div>]
*/
print(soup.find_all("div",attrs={'float':'left'}))		//[<div class="info" float="left">welcome to xupt</div>]
print('-------------------css()--------------------')
print(soup.select('title'))	//[<title id="title">西邮</title>]
print(soup.select('#title'))	//[<title id="title">西邮</title>]
print(soup.select('.info'))
/*
[<div class="info" float="left">welcome to xupt</div>, <div class="info" float="right">
<span>Good Study</span>
<a href="www.baidu.com"></a>
<strong><!--注释!--></strong>
</div>]
*/
print(soup.select('div span'))	//[<span>Good Study</span>]
print(soup.select('div > span'))	//[<span>Good Study</span>]
print(soup.select('div')[1].select('a'))	//[<a href="www.baidu.com"></a>]
print(soup.select('title')[0].text)	//西邮
案例

个人博客整理案例

import requests
from bs4 import BeautifulSoup
import re


def get_content(url,):
    try:
        user_agent = "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/59.0.3071.109 Safari/537.36"
        response = requests.get(url,  headers={'User-Agent': user_agent})
        response.raise_for_status()   # 如果返回的状态码不是200, 则抛出异常;
        response.encoding = response.apparent_encoding  # 判断网页的编码格式, 便于respons.text知道如何解码;
    except Exception as e:
        print("爬取错误")
    else:

        print(response.url)
        print("爬取成功!")
        return  response.content


def parser_content(htmlContent):
    # 实例化soup对象, 便于处理;
    soup = BeautifulSoup(htmlContent, 'html.parser')
    # 1). 获取每个博客的大盒子: 特征: div标签, class名称一致article-item-box csdn-tracking-statistics
    #  <div class="article-item-box csdn-tracking-statistics" data-articleid="85718923">
    divObjs = soup.find_all('div', class_="article-item-box")
    # 2). 依次遍历每一个div标签, 获取博客标题
    #  博客标题的特征: h4里面的a标签里面的内容
    # 去掉默认的广告, 留下个人的博客内容;

    for  divObj in divObjs[1:]:
        # **2-1. 获取博客标题: 去掉原创或者转载的信息, 只需要博客名称;
        title = divObj.h4.a.get_text().split()[1]
        # **2-2. 获取博客链接, 也就是获取a链接中href对应的值;
        blogUrl = divObj.h4.a.get('href')
        global  bloginfo
        # 将爬取的所有内容保存到变量中[(blogtitle, blogurl)]
        bloginfo.append((title, blogUrl))


if __name__ == '__main__':
    blogPage = 3
    # 全局变量, 用于保存所有博客信息;
    bloginfo = []
    for page in range(1, blogPage+1):
        url = "https://blog.csdn.net/King15229085063/article/list/%s" %(page)
        content = get_content(url)
        parser_content(content)
        print("第%d页整理结束...." %(page))


    with open('doc/myblog.md', 'a') as f:
        for index, info in enumerate(bloginfo[::-1]):
            f.write('- 第%d篇博客: [%s](%s)\n' %(index+1, info[0], info[1]))
    print("完成.....")

爬取豆瓣 TOP250电影信息

"""
url = "http://movie.douban.com/top250/"

需要获取的信息: 电影名称, 电影评分, 评价人数, 电影短评
"""
import re

import requests
from bs4 import BeautifulSoup

def get_content(url,):
    try:
        user_agent = "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/59.0.3071.109 Safari/537.36"
        response = requests.get(url,  headers={'User-Agent': user_agent})
        response.raise_for_status()   # 如果返回的状态码不是200, 则抛出异常;
        response.encoding = response.apparent_encoding  # 判断网页的编码格式, 便于respons.text知道如何解码;
    except Exception as e:
        print("爬取错误")
    else:

        print(response.url)
        print("爬取成功!")
        return  response.content

def parser_content(htmlContent):
    # 实例化soup对象, 便于处理;
    soup = BeautifulSoup(htmlContent, 'html.parser')
    #  1). 电影信息存储在ol标签里面的li标签:
    #  <ol class="grid_view">
    olObj = soup.find_all('ol', class_='grid_view')[0]

    #  2). 获取每个电影的详细信息, 存储在li标签;
    details = olObj.find_all('li')


    for detail in details:
        # #  3). 获取电影名称;
        movieName = detail.find('span', class_='title').get_text()

        # 4). 电影评分:
        movieScore = detail.find('span', class_='rating_num').get_text()

        # 5). 评价人数***************
        # 必须要转换类型为字符串
        movieCommentNum = str(detail.find(text=re.compile('\d+人评价')))


        # 6). 电影短评
        movieCommentObj = detail.find('span', class_='inq')
        if movieCommentObj:
            movieComment = movieCommentObj.get_text()
        else:
            movieComment = "无短评"

        movieInfo.append((movieName, movieScore, movieCommentNum, movieComment))




import openpyxl


def create_to_excel(wbname, data, sheetname='Sheet1', ):
    """
    将制定的信息保存到新建的excel表格中;

    :param wbname:
    :param data: 往excel中存储的数据;
    :param sheetname:
    :return:
    """

    print("正在创建excel表格%s......" % (wbname))

    # wb = openpyxl.load_workbook(wbname)
    #  如果文件不存在, 自己实例化一个WorkBook的对象;
    wb = openpyxl.Workbook()
    # 获取当前活动工作表的对象
    sheet = wb.active
    # 修改工作表的名称
    sheet.title = sheetname
    # 将数据data写入excel表格中;
    print("正在写入数据........")
    for row, item in enumerate(data):  # data发现有4行数据, item里面有三列数据;
        print(item)
        for column, cellValue in enumerate(item):

            # cell = sheet.cell(row=row + 1, column=column + 1, value=cellValue)
            cell = sheet.cell(row=row+1, column=column + 1)
            cell.value = cellValue

    wb.save(wbname)
    print("保存工作薄%s成功......." % (wbname))


if __name__ == '__main__':
    doubanTopPage = 2
    perPage = 25
    # [(), (), ()]
    movieInfo = []
    # 1, 2, 3 ,4, 5
    for page in range(1, doubanTopPage+1):
        # start的值= (当前页-1)*每页显示的数量(25)
        url = "https://movie.douban.com/top250?start=%s" %((page-1)*perPage)
        content = get_content(url)
        parser_content(content)


    create_to_excel('/tmp/hello.xlsx', movieInfo, sheetname="豆瓣电影信息")


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值