python爬虫之网络爬虫概述/请求模块/正则解析

网络爬虫概述

  • 定义

网络蜘蛛、网络机器人,抓取网络数据的程序
其实就是用Python程序模仿人点击浏览器并访问网站,而且模仿的越像越好,让Web站点无法发现你不是人
  • 爬取数据目的

1、公司项目测试数据
2、公司业务部门及其他部门所需数据
3、数据分析
  • 企业获取数据方式

1、公司自有数据
2、第三方数据平台购买(数据堂、贵阳大数据交易所)
3、爬虫爬取数据
  • Python做爬虫优势

1、Python :请求模块、解析模块丰富成熟,强大的Scrapy网络爬虫框架
2、PHP :对多线程、异步支持不太好
3、JAVA:代码笨重,代码量大
4、C/C++:虽然效率高,但是代码成型慢
  • 爬虫分类

1、通用网络爬虫(搜索引擎使用,遵守robots协议)
  robots协议 :网站通过robots协议告诉搜索引擎哪些页面可以抓取,哪些页面不能抓取,通用网络爬虫需要遵守robots协议(君子协议)
	https://www.taobao.com/robots.txt
2、聚焦网络爬虫 :自己写的爬虫程序
  • 爬虫爬取数据步骤

1、确定需要爬取的URL地址
2、由请求模块向URL地址发出请求,并得到网站的响应
3、从响应内容中提取所需数据
   1、所需数据,保存
   2、页面中有其他需要继续跟进的URL地址,继续第2步去发请求,如此循环

爬虫请求模块一

模块名及导入

1、模块名:urllib.request
2、导入方式:
   1import urllib.request
   2from urllib import request

常用方法详解

urllib.request.urlopen
  • 作用

向网站发起请求并获取响应对象

  • 参数
1、url:需要爬取的URL地址
2、timeout: 设置等待超时时间,指定时间内未得到响应抛出超时异常
  • 第一个爬虫程序 - 01_urlopen.py

打开浏览器,输入百度地址(http://www.baidu.com/),得到百度的响应

import urllib.request

# urlopen() : 向URL发请求,返回响应对象
response=urllib.request.urlopen('http://www.baidu.com/')
# 提取响应内容
html = response.read().decode('utf-8')
# 打印响应内容
print(html)
  • 响应对象(response)方法
1bytes = response.read() # read()得到结果为 bytes 数据类型
2、string = response.read().decode() # decode() 转为 string 数据类型
3、url = response.geturl() # 返回实际数据的URL地址
4、code = response.getcode() # 返回HTTP响应码
# 补充
5、string.encode() # bytes -> string
6bytes.decode()  # string -> bytes

思考:网站如何来判定是人类正常访问还是爬虫程序访问???

# 向测试网站: http://httpbin.org/get 发请求,查看自己请求头 - 响应内容
# 代码如下

此处各位大佬自己完成

# html中的请求头headers如下
"headers": {
    "Accept-Encoding": "identity", 
    "Host": "httpbin.org", 
    "User-Agent": "Python-urllib/3.6"
  }, 
发现请求头中User-Agent竟然是:Python-urllib/3.6!!!!!!!!!!!!!!!!!!!
我们需要重构User-Agent,发请求时带着User-Agent过去,但是 urlopen()方法不支持重构User-Agent,那我们怎么办?请看下面的方法!!!
urllib.request.Request
  • 作用

创建请求对象(包装请求,重构User-Agent,使程序更像正常人类请求)

  • 参数
1、url:请求的URL地址
2、headers:添加请求头(爬虫和反爬虫斗争的第一步)
  • 使用流程
1、构造请求对象(重构User-Agent)
2、发请求获取响应对象(urlopen)
3、获取响应对象内容
  • 示例 - 02_Request.py

向测试网站(http://httpbin.org/get)发起请求,构造请求头并从响应中确认请求头信息

import urllib.request

url = "http://httpbin.org/get"

# 1.创建请求对象
req = urllib.request.Request(
    url=url,
    headers={"User-Agent":'Mozilla/5.0'}
)
# 2.发请求获取相应对象
res = urllib.request.urlopen(req)
# 3.提取响应对象内容
html = res.read().decode()
print(html)

URL地址编码模块

模块名及导入

  • 模块
# 模块名
urllib.parse

# 导入
import urllib.parse
from urllib import parse
  • 作用

给URL地址中查询参数进行编码

编码前:https://www.baidu.com/s?wd=美女  #浏览器解码为中文
编码后:https://www.baidu.com/s?wd=%E7%BE%8E%E5%A5%B3

常用方法

urllib.parse.urlencode({dict})
  • URL地址中一个查询参数
# 查询参数:{'wd' : '美女'}
# urlencode编码后:'wd=%e7%be%8e%e5%a5%b3'

# 示例代码
query_string = {'wd' : '美女'}
result = urllib.parse.urlencode(query_string)
# result: 'wd=%e7%be%8e%e5%a5%b3'
  • URL地址中多个查询参数
from urllib import parse
params = {
	'wd' : '美女',
	'pn' : '50'
}
params = parse.urlencode(query_string_dict)
url = 'http://www.baidu.com/s?{}'.format(params)
print(url)
  • 拼接URL地址的3种方式
# 1、字符串相加
  baseurl = 'http://www.baidu.com/s?'
  params = 'wd=%E7XXXX&pn=20'
  url = baseurl + params

# 2、字符串格式化(占位符)
  params = 'wd=%E7XXXX&pn=20'
  url = 'http://www.baidu.com/s?%s'% params

# 3、format()方法
  url = 'http://www.baidu.com/s?{}'
  params = 'wd=#E7XXXX&pn=20'
  url = url.format(params)
  • 练习
    在百度中输入要搜索的内容,把响应内容保存到本地文件
请输入搜索内容: 赵丽颖
# 最终保存到本地文件 - 赵丽颖.html

代码实现 - 03_parse_baidu.py

"""
在百度中输入要搜索的内容,把响应内容保存到本地文件
"""
import urllib
from urllib import parse, request

# 1.拼url地址
url = "http://www.baidu.com/s?{}"
word = input("请输入搜索内容:")
url_str = url.format(parse.urlencode({"wd": word}))

# 2.发请求存到本地
req = request.Request(url=url_str, headers={"User-Agent": "Mozilla/5.0"})
res = urllib.request.urlopen(req)
html = res.read().decode()
filename=word+".txt"
with open(filename,"w+") as fp:
    fp.write(html)


quote(string)编码
  • 示例1
from urllib import parse

string = '美女'
print(parse.quote(string))
# 结果: %E7%BE%8E%E5%A5%B3

改写之前urlencode()代码,使用quote()方法实现

from urllib import parse

url = 'http://www.baidu.com/s?wd={}'
word = input('请输入要搜索的内容:')
query_string = parse.quote(word)
print(url.format(query_string))
unquote(string)解码
  • 示例
from urllib import parse

string = '%E7%BE%8E%E5%A5%B3'
result = parse.unquote(string)
print(result)
总结
# 1、urllib.request
req=urllib.request.Request(url,headers)
res=urllib.request.urlopen(req)

# 2、响应对象res方法
html=res.read().decode()
url=res.geturl()
code=res.code()

# 3、urllib.parse(编码)
parms=urllib.parse.urlencode({"wd":"name"})
parms=urllib.parse.quote("name")

百度贴吧数据抓取案例

要求

1、输入贴吧名称:赵丽颖吧
2、输入起始页:1
3、输入终止页:3
4、保存到本地文件
   赵丽颖吧-1.html、赵丽颖吧-2.html ...

实现步骤

  • 1、查看是否为静态页面
右键 - 查看网页源代码 - 搜索数据关键字
  • 2、找URL规律
1:http://tieba.baidu.com/f?kw=??&pn=02:http://tieba.baidu.com/f?kw=??&pn=50
第n页:pn=(n-1)*50
  • 3、获取网页内容

  • 4、提取所需数据

  • 5、保存(本地文件、数据库)

    代码实现 - 04_tieba_spider.py

    import random
    from urllib import request, parse
    import time
    
    from day01.User_Agent import getheaders
    
    
    class TieBaSpiser:
        def __init__(self):
            self.url = "http://tieba.baidu.com/f?kw={}&pn={}"
            self.headers = getheaders()
    
        # 请求
        def request(self, kw, pn):
            req = request.Request(url=self.url.format(parse.quote(kw),parse.quote(pn)), headers=self.headers)
            res = request.urlopen(req)
            html = res.read().decode('utf-8','ignore')
            return html
    
        # 解析
        def parse_html(self):
            pass
    
        # 保存
        def save_html(self, filename, html):
            with open(filename, "w") as f:
                f.write(html)
    
        # 函数
        def run(self):
            name = input("请输入贴吧名:")
            begin = int(input("请输入起始页:"))
            stop = int(input("请输入终止页:"))
            # 拼接地址
            for page in range(begin, stop + 1):
                pn = (page - 1) * 50
                html = self.request(name, str(pn))
                filename = "%s-第%d页.html" % (name, page)
                self.save_html(filename, html)
                # 提示
                print("第%d页爬取成功" % page)
                # 每爬取一个页面休眠1-3秒
                time.sleep(random.randint(1, 3))
    
    
    if __name__ == '__main__':
        start = time.time()
        spiser = TieBaSpiser()
        spiser.run()
        end = time.time()
        print("执行时间:%.2f" % (end - start))
    
    

正则解析模块

re模块使用流程

  • 方法一
   r_list=re.findall('正则表达式',html,re.S)
  • 方法二
# 1、创建正则编译对象
pattern = re.compile(r'正则表达式',re.S)
r_list = pattern.findall(html)

正则表达式元字符

元字符含义
.任意一个字符(不包括\n)
\d一个数字
\s空白字符
\S非空白字符
[]包含[]内容
*出现0次或多次
+出现1次或多次

思考:请写出匹配任意一个字符的正则表达式?

import re
# 方法一
pattern = re.compile('.',re.S)
# 方法二
pattern = re.compile('[\s\S]')

贪婪匹配和非贪婪匹配

  • 贪婪匹配(默认)
1、在整个表达式匹配成功的前提下,尽可能多的匹配 * + ?
2、表示方式: .* .+ .?
  • 非贪婪匹配
1、在整个表达式匹配成功的前提下,尽可能少的匹配 * + ?
2、表示方式:.*? .+?

示例代码 - 05_re_greed.py

import re


html="""
<div><p>陈志刚</p></div>
<div><p>PHP</p></div>
"""

#贪婪匹配
pattern=re.compile("<div><p>.*</p></div>",re.S)
r=pattern.findall(html)
#['<div><p>陈志刚</p></div>\n<div><p>PHP</p></div>']
print(r)
#非贪婪匹配
pattern=re.compile("<div><p>.*?</p></div>",re.S)
r=pattern.findall(html)
#['<div><p>陈志刚</p></div>', '<div><p>PHP</p></div>']
print(r)

#内容匹配
pattern=re.compile("<div><p>(.*?)</p></div>",re.S)
r=pattern.findall(html)
#['陈志刚', 'PHP']
print(r)

正则表达式分组

  • 作用

在完整的模式中定义子模式,将每个圆括号中子模式匹配出来的结果提取出来

  • 示例
import re

s = 'A B C D'
p1 = re.compile('\w+\s+\w+')
print(p1.findall(s))
# 结果: ???["A B","C D"]

p2 = re.compile('(\w+)\s+\w+')
print(p2.findall(s))
# 结果: ???
#第一步:整体正则:["A B","C D"]
#第二步:提取分组:["A","C"]

p3 = re.compile('(\w+)\s+(\w+)')
print(p3.findall(s))
# 结果: ???
#第一步:整体正则:["A B","C D"]
#第二步:提取分组:[("A","B"),("C","D")]
  • 分组总结
1、在网页中,想要什么内容,就加()
2、先按整体正则匹配,然后再提取分组()中的内容
  如果有2个及以上分组(),则结果中以元组形式显示 [('小区1','500万'),('小区2','600万'),()]
  • 练习

页面结构如下:

# <div class="animal">.*?title="(.*?)".*?
<div class="animal">
    <p class="name">
		<a title="Tiger"></a>
    </p>
    <p class="content">
		Two tigers two tigers run fast
    </p>
</div>

<div class="animal">
    <p class="name">
		<a title="Rabbit"></a>
    </p>

    <p class="content">
		Small white rabbit white and white
    </p>
</div>

从以上html代码结构中完成如下内容信息的提取:

# 问题1
[('Tiger',' Two...'),('Rabbit','Small..')]
# 问题2
动物名称 :Tiger
动物描述 :Two tigers two tigers run fast
***************************************
动物名称 :Rabbit
动物描述 :Small white rabbit white and white

代码实现 - 06_re_exercise.py

import re

html="""
<div class="animal">
    <p class="name">
		<a title="Tiger"></a>
    </p>
    <p class="content">
		Two tigers two tigers run fast
    </p>
</div>

<div class="animal">
    <p class="name">
		<a title="Rabbit"></a>
    </p>

    <p class="content">
		Small white rabbit white and white
    </p>
</div>
"""
# pattern=re.compile(r'<div class="animal">.*?title="(.*?)".*?"content">\n\t\t(.*?)\n',re.S)
pattern=re.compile(r'<div class="animal">.*?title="(.*?)".*?"content">(.*?)</p>',re.S)
result=pattern.findall(html)
for i in range(len(result)):
    print("动物名称:",result[i][0])
    print("动物描述:",result[i][1].strip())

今日任务

1、把百度贴吧案例重写一遍,不要参照课上代码
2、爬取猫眼电影信息 :猫眼电影-榜单-top100榜

1步完成:
	猫眼电影-1.html
	猫眼电影-2.html
	... ...2步完成:
	1、提取数据 :电影名称、主演、上映时间
	2、先打印输出,然后再写入到本地文件

3、复习任务

pymysql、MySQL基本命令
MySQL :建库建表普通查询等
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值