写一个最简单的爬虫程序

Write the simplest crawler program

Posted by Zhihao on 2019-05-20

The First Title Picture

本文将介绍如何写一个最简单的Python爬虫程序

主要分为以下 5 部分内容:

  1. 了解网页基础;
  2. 使用 requests 库抓取网站数据;
  3. 使用 Beautiful Soup 解析网页;
  4. 清洗和组织数据;
  5. 爬虫攻防战;

1 网页基础

1.1 网页结构

网页一般由三部分组成,分别是 HTML(超文本标记语言)、CSS(层叠样式表)和 JScript(活动脚本语言)。

HTML

HTML是整个网页的结构,相当于整个网站的框架。带“<”和“>”符号的都是属于HTML的标签,并且标签都是成对出现的。

常见的标签如下:

1
2
3
4
5
6
7
8
<html>..</html> 表示标记中间的元素是网页
<body>..</body> 表示用户可见的内容
<div>..</div> 表示框架
<p>..</p> 表示段落
<li>..</li>表示列表
<img>..</img>表示图片
<h1>..</h1>表示标题
<a href="">..</a>表示超链接

CSS

CSS 表示样式,比如标签中的<style type="text/css">表示下面引用一个 CSS,在 CSS中定义了外观。

JScript

JScript表示功能。交互的内容和各种特效都在JScript中,JScript 描述了网站中的各种功能。

如果用人体来比喻,HTML 是人的骨架,并且定义了人的嘴巴、眼睛、耳朵等要长在哪里。CSS 是人的外观细节,如嘴巴长什么样子,眼睛是双眼皮还是单眼皮,是大眼睛还是小眼睛,皮肤是黑色的还是白色的等。JScript 表示人的技能,例如跳舞、唱歌或者演奏乐器等。

1.2 写一个简单的 HTML

通过编写和修改HTML,可以更好地理解HTML。首先打开一个记事本,然后输入下面的内容:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
<html>
<head>
<title> Python 3 爬虫与数据清洗入门与实战</title>
</head>
<body>
<div>
<p>Python 3爬虫与数据清洗入门与实战</p>
</div>
<div>
<ul>
<li><a href="http://c.biancheng.net">爬虫</a></li>
<li>数据清洗</li>
</ul>
</div>
</body>

输入代码后,保存记事本,然后修改文件名和后缀名为”HTML.html”;

用浏览器打开后的效果,如下图所示:

The First Title Picture

这段代码只是用到了 HTML,读者可以自行修改代码中的中文,然后观察其变化。

1.3 关于爬虫的合法性

几乎每一个网站都有一个名为 robots.txt 的文档,当然也有部分网站没有设定 robots.txt。对于没有设定 robots.txt 的网站可以通过网络爬虫获取没有口令加密的数据,也就是该网站所有页面数据都可以爬取。如果网站有 robots.txt 文档,就要判断是否有禁止访客获取的数据。

以淘宝网为例,在浏览器中访问 https://www.taobao.com/robots.txt,如下图所示:

The First Title Picture

淘宝网允许部分爬虫访问它的部分路径,而对于没有得到允许的用户,则全部禁止爬取,代码如下:

1
2
User-Agent:*
Disallow:/

这一句代码的意思是除前面指定的爬虫外,不允许其他爬虫爬取任何数据。

2 使用 requests 库请求网站

2.1 安装 requests 库

终端输入命令pip install requests,就可以开始安装了。

2.2 爬虫的基本原理

网页请求的过程分为两个环节:

  1. Request (请求):每一个展示在用户面前的网页都必须经过这一步,也就是向服务器发送访问请求。

  2. Response(响应):服务器在接收到用户的请求后,会验证请求的有效性,然后向用户(客户端)发送响应的内容,客户端接收服务器响应的内容,将内容展示出来,就是我们所熟悉的网页请求,如下图所示。

    The First Title Picture

网页请求的方式也分为两种:

  1. GET:最常见的方式,一般用于获取或者查询资源信息,也是大多数网站使用的方式,响应速度快。
  2. POST:相比 GET 方式,多了以表单形式上传参数的功能,因此除查询信息外,还可以修改信息。

所以,在写爬虫前要先确定网页请求的方式是什么。

2.3 网页请求方式一:使用 GET 方式抓取数据

随意打开一个网站,比如下图所示的某瓣,在源码中【Ctrl+F】搜索页面出现的字词,比如“正在热映”四个字,如果能在源码中直接搜索到,那么这个网站的请求方式就是GET。如下图所示:

The First Title Picture

确定好请求对象和方式后,输入以下代码:

1
2
3
4
import requests                    #导入requests包
url = 'http://www.cntour.cn/'
strhtml = requests.get(url) #Get方式获取网页数据
print(strhtml.text)

如果成功的话,你已经看到strhtml.text打印的结果。

接下来简单介绍一下上面几行代码的意思:

1
import requests  #加载库使用的语句是 import+库的名字

GET 方式获取数据需要调用 requests 库中的 get 方法,使用方法是在 requests 后输入英文点号,如下所示:

1
requests.get

将获取到的数据存到 strhtml 变量中,代码如下:

1
strhtml = request.get(url)

这个时候 strhtml 是一个 URL 对象,它代表整个网页,但此时只需要网页中的源码,下面的语句表示网页源码:

1
strhtml.text

2.4 网页请求方式二:使用 POST 方式抓取数据

首先输入有道翻译的网址:http://fanyi.youdao.com/,进入有道翻译页面。
按快捷键 F12,进入开发者模式,单击 Network,此时内容为空,如下图所示:

The First Title Picture

在有道翻译中输入“我爱中国”,单击“翻译”按钮,如下图所示:

The First Title Picture

在开发者模式中,依次单击Network按钮和XHR按钮,找到翻译数据,如下图所示:

The First Title Picture

单击 Headers,发现请求数据的方式为 POST。如下图所示:

The First Title Picture

找到数据所在之处并且明确请求方式之后,接下来开始撰写爬虫。
首先,将 Headers 中的 URL 复制出来,并赋值给 url,代码如下:

1
url = 'http://fanyi.youdao.com/translate_o?smartresult=dict&smartresult=rule'

POST 的请求获取数据的方式不同于 GETPOST 请求数据必须构建请求头才可以。
Form Data 中的请求参数如下图所示:

The First Title Picture

将其复制并构建一个新字典:

1
From_data={'i':'我愛中國','from':'zh-CHS','to':'en','smartresult':'dict','client':'fanyideskweb','salt':'15477056211258','sign':'b3589f32c38bc9e3876a570b8a992604','ts':'1547705621125','bv':'b33a2f3f9d09bde064c9275bcb33d94e','doctype':'json','version':'2.1','keyfrom':'fanyi.web','action':'FY_BY_REALTIME','typoResult':'false'}

接下来使用 requests.post 方法请求表单数据,代码如下:

1
2
import requests            #导入requests包
response = requests.post(url,data=payload)

将字符串格式的数据转换成JSON 格式数据,并根据数据结构,提取数据,并将翻译结果打印出来,代码如下:

1
2
3
import json
content = json.loads(response.text)
print(content['translateResult'][0][0]['tgt'])

使用 requests.post 方法抓取有道翻译结果的完整代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
import requests        #导入requests包
import json
def get_translate_date(word=None):
url = 'http://fanyi.youdao.com/translate_o?smartresult=dict&smartresult=rule'
From_data={'i':word,'from':'zh- CHS','to':'en','smartresult':'dict','client':'fanyideskweb','salt':'15477056211258','sign':'b3589f32c38bc9e3876a570b8a992604','ts':'1547705621125','bv':'b33a2f3f9d09bde064c9275bcb33d94e','doctype':'json','version':'2.1','keyfrom':'fanyi.web','action':'FY_BY_REALTIME','typoResult':'false'}
#请求表单数据
response = requests.post(url,data=From_data)
#将Json格式字符串转字典
content = json.loads(response.text)
print(content)
#打印翻译后的数据
#print(content['translateResult'][0][0]['tgt'])
if __name__=='__main__':
get_translate_date('我爱中国')

3 使用 Beautiful Soup 解析网页

通过 requests 库已经可以抓到网页源码,接下来要从源码中找到并提取数据。Beautiful Souppython 的一个库,其最主要的功能是从网页中抓取数据。Beautiful Soup 目前已经被移植到 bs4 库中,也就是说在导入 Beautiful Soup 时需要先安装bs4 库。

终端直接输入pip3 install beautifulsoup4安装。

安装好 bs4 库以后,还需安装lxml 库。如果我们不安装 lxml库,就会使用 Python 默认的解析器。尽管 Beautiful Soup既支持Python 标准库中的HTML解析器又支持一些第三方解析器,但是 lxml 库具有功能更加强大、速度更快的特点,因此笔者推荐安装lxml库。

安装 Python第三方库后,输入下面的代码,即可开启Beautiful Soup之旅:

1
2
3
4
5
6
7
import requests                           #导入requests包
from bs4 import BeautifulSoup
url='http://www.cntour.cn/'
strhtml=requests.get(url)
soup=BeautifulSoup(strhtml.text,'lxml')
data = soup.select('#main>div>div.mtop.firstMod.clearfix>div.centerBox>ul.newsList>li>a')
print(data)

Beautiful Soup 库能够轻松解析网页信息,它被集成在 bs4 库中,需要时可以从 bs4 库中调用。其表达语句如下:

1
from bs4 import BeautifulSoup

首先,HTML 文档将被转换成 Unicode 编码格式,然后 Beautiful Soup 选择最合适的解析器来解析这段文档,此处指定 lxml解析器进行解析。解析后便将复杂的 HTML 文档转换成树形结构,并且每个节点都是 Python 对象。这里将解析后的文档存储到新建的变量 soup 中,代码如下:

1
soup=BeautifulSoup(strhtml.text,'lxml')

接下来用 select(选择器)定位数据,定位数据时需要使用浏览器的开发者模式,将鼠标光标停留在对应的数据位置并右击,然后在快捷菜单中选择检查命令,如下图所示:

The First Title Picture

随后在浏览器右侧会弹出开发者界面,右侧高亮的代码(下图(b))对应着左侧高亮的数据文本(下图(a))。右击右侧高亮数据,在弹出的快捷菜单中选择“Copy”➔“Copy Selector”命令,便可以自动复制路径。

The First Title Picture

将路径粘贴在文档中,代码如下:

1
#main > div > div.mtop.firstMod.clearfix > div.centerBox > ul.newsList > li:nth-child(1) > a

由于这条路径是选中的第一条的路径,而我们需要获取所有的头条新闻,因此将 li:nth-child(1)中冒号(包含冒号)后面的部分删掉,代码如下:

1
#main > div > div.mtop.firstMod.clearfix > div.centerBox > ul.newsList > li > a

使用 soup.select 引用这个路径,代码如下:

1
data = soup.select('#main > div > div.mtop.firstMod.clearfix > div.centerBox > ul.newsList > li > a')

4 清洗和组织数据

至此,获得了一段目标的 HTML代码,但还没有把数据提取出来,输入以下代码:

1
2
3
4
5
6
for item in data:
result={
'title':item.get_text(),
'link':item.get('href')
}
print(result)

代码运行结果如下图所示:

The First Title Picture

首先明确要提取的数据是标题和链接,标题在<a>标签中,提取标签的正文用 get_text() 方法。链接在<a>标签的href属性中,提取标签中的href属性用 get()方法,在括号中指定要提取的属性数据,即get('href')

从上图中可以发现,文章的链接中有一个数字 ID。下面用正则表达式提取这个 ID。需要使用的正则符号如下:

1
2
\d匹配数字
+匹配前一个字符1次或多次

在 Python 中调用正则表达式时使用 re 库,这个库不用安装,可以直接调用。输入以下代码:

1
2
3
4
5
6
7
8
import re
for item in data:
result={
"title":item.get_text(),
"link":item.get('href'),
'ID':re.findall('\d+',item.get('href'))
}
print(result)

运行结果如下图所示:

The First Title Picture

这里使用 re 库的 findall 方法,第一个参数表示正则表达式,第二个参数表示要提取的文本。

5 爬虫攻防战

爬虫是模拟人的浏览访问行为,进行数据的批量抓取。当抓取的数据量逐渐增大时,会给被访问的服务器造成很大的压力,甚至有可能崩溃。换句话就是说,服务器是不喜欢有人抓取自己的数据的。那么,网站方面就会针对这些爬虫者,采取一些反爬策略。

服务器第一种识别爬虫的方式就是通过检查连接的 useragent 来识别到底是浏览器访问,还是代码访问的。如果是代码访问的话,访问量增大时,服务器会直接封掉来访 IP。

那么应对这种初级的反爬机制,我们应该采取何种举措?

还是以前面创建好的爬虫为例。在进行访问时,我们在开发者环境下不仅可以找到 URLForm Data,还可以在 Request headers 中构造浏览器的请求头,封装自己。服务器识别浏览器访问的方法就是判断 keyword 是否为 Request headers 下的 User-Agent,如所示。

The First Title Picture

因此,我们只需要构造这个请求头的参数。创建请求头部信息即可,代码如下:

1
2
headers={'User-Agent':'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/70.0.3538.110 Safari/537.36'}
response = request.get(url,headers=headers)

在实际操作中加了头部信息之后,还是会出现urllib.error.HTTPError: HTTP Error 403: Forbidden的问题,开审查元素,到Network界面刷新,发现图片加载时有Refererheader,加上Referer之后,问题解决。

1
2
3
opener=urllib.request.build_opener()
opener.addheaders=[('User-Agent','Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.93 Safari/537.36'),('Referer',m_url)]
urllib.request.install_opener(opener)

写到这里,很多读者会认为修改 User-Agent 很太简单。确实很简单,但是正常人1秒看一个图,而个爬虫1秒可以抓取好多张图,比如 1 秒抓取上百张图,那么服务器的压力必然会增大。也就是说,如果在一个 IP 下批量访问下载图片,这个行为不符合正常人类的行为,肯定要被封 IP。

其原理也很简单,就是统计每个IP的访问频率,该频率超过阈值,就会返回一个验证码,如果真的是用户访问的话,用户就会填写,然后继续访问,如果是代码访问的话,就会被封 IP。

这个问题的解决方案有两个,第一个就是常用的增设延时,每 3 秒钟抓取一次,代码如下:

1
2
import time
time.sleep(3)

但是,我们写爬虫的目的是为了高效批量抓取数据,这里设置 3 秒钟抓取一次,效率未免太低。其实,还有一个更重要的解决办法,那就是从本质上解决问题。

不管如何访问,服务器的目的就是查出哪些为代码访问,然后封锁 IP。解决办法:为避免被封 IP,在数据采集时经常会使用代理。当然,requests 也有相应的 proxies属性。

首先,构建自己的代理 IP 池,将其以字典的形式赋值给 proxies,然后传输给 requests,代码如下:

1
2
3
4
5
proxies={
"http":"http://10.10.1.10:3128",
"https":"http://10.10.1.10:1080",
}
response = requests.get(url, proxies=proxies)

下面为一段实际抓取图片的代码供参考:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
#coding=utf-8
import requests #导入requests包
from bs4 import BeautifulSoup
import time
import re
import urllib.request


def get_img(m_url,pic_name):
strhtml = requests.get(url=m_url) #Get方式获取网页数据
soup=BeautifulSoup(strhtml.text,'lxml')
data = soup.select('xxxxx') #用BeautifulSoup清晰数据

for item in data:
result = {'link':item.get('src'),'title':item.get('title')} #提取元素
img_url = result['link'] #获取图片链接
img_name = pic_name + ' ' + result['title'] + '.jpg' #获取图片的名字


opener=urllib.request.build_opener()
opener.addheaders=[('User-Agent','Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/70.0.3538.110 Safari/537.36'),('Referer',m_url)]
urllib.request.install_opener(opener)
urllib.request.urlretrieve(img_url,'下载路径')
print('成功下载'+img_name)

def main():
m_url = 'xxxxx'
pic_name = 'xxxxx'
get_img(m_url,pic_name)
time.sleep(5) #程序停顿5秒,此处即可实现批量化程序

if __name__ == '__main__':
main()

6 其他

此文大量参照此博客内容,并加入自己的内容以及内容抓取相关介绍。


...

...

00:00
00:00