爬虫实践之XX行代码爬取10000菜谱数据

什么是爬虫爬虫:又叫做网络蜘蛛,是一段自动抓取互联网信息的程序,从互联网上抓取对于我们有价值的信息。

如何合法地爬虫

有些网站不允许网络爬虫,或是对可爬取的内容做了限制,一个网站的爬虫协议可通过访问该网站的robots.txt文件获得

以豆瓣网为例

可以看到,豆瓣对于不同的访问者有不同的访问限制,其中对于用户名为 Wandoujia Spider的访问者,豆瓣不允许访问。

我用到的菜谱网站对爬虫无限制,所以爬虫是合法的。

本篇特色

连续爬取10000个网页

引入第三方库

import requests #发送请求

import re #正则表达式,用于提取网页数据

import winsound #提醒程序运行结束

import time #计算程序运行时间

如果没有安装这些第三方库,可以在命令提示符中输入如下代码,进行下载

pip install requests,re,winsound,time

爬虫的三个步骤获取要爬取的所有网页的网址

提取网页内容中的有用信息

信息导出

每个步骤对应一个函数

Step_1 获取要爬取的所有网页的网址

首先我们打开该网址,查看它的源代码网页源代码

观察发现每道菜对应的网址在这个目录下

用正则表达式获得该网址,写入列表中

由于每一个网页只有十道菜,点击下一页后发现网页的变化规律为换页时网址只有数字改变

可以看到最多有1000页,也就是有10000道菜

使用循环,将每一页的菜对应的网址都写入列表,每一次写入时写入一行列表,多次写入后,形成一个二维的列表,前两页的菜谱网址在二维列表中显示如下:

代码如下

all_url = [] #创建一个数组用于存储网页地址

def get_all_url(n): #这个函数用于获得网页中的菜的全部网址

if(n==1):

url = "https://m.meishij.net/caixi/sucai1/"

else:

url='https://m.meishij.net/caixi/sucai1/p%s/'%n #%s相当于C语言中的%s,表示格式化一个对象为字符,同理%d表示格式化一个对象为整数

headers = {"User-Agent":"Mozilla/5.0 (Windows NT 10.0; Win64; x64) \

AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.122 Safari/537.36" }

response = requests.get(url,headers=headers) #访问网页

response.encoding = "utf-8" #设置接收编码格式

pattern = re.compile(r'', re.S)

#正则表达式提取网页中的网址,re.S表示在整个文本中进行匹配,如果不加re.S,将只在一行进行匹配

result = pattern.findall(response.text) #获取的网页结果存储到result里

all_url.append(result[0:10])#由于每页只有十道菜,result中只有前十条对应的是菜的网址,故我们只添加前十条

return all_url #作为返回值返回这个列表

关于headers的说明

在使用python爬虫爬取数据的时候,经常会遇到一些网站的反爬虫措施,一般就是针对于headers中的User-Agent,如果没有对headers进行设置,User-Agent会声明自己是python脚本,而如果网站有反爬虫的想法的话,必然会拒绝这样的连接。而修改headers可以将自己的爬虫脚本伪装成浏览器的正常访问,来避免这一问题。 点击这里了解详情

关于编码格式utf-8 的说明

utf-8,可以编码中文,大部分python编译器都默认编码方式为utf-8 点击这里了解详情

Step_2 提取网页中的有用信息

打开一道菜的网址,查看源代码,寻找我们需要的信息在哪里,然后用正则表达式获取,流程与上个函数获取网址相同

主辅料信息在这里

特征信息在这里(包括做法和口味)

def get_info(resp,output):

name_pattern = re.compile(r'

(.*)

')# 正则表达式获取菜名信息

food_pattern = re.compile(r'(.*)(.*)

')# 正则表达式获得主料信息

fixing_pattern = re.compile(r'

(.*) (.*)
') # 正则表达式获得辅料信息

fearture1_pattern = re.compile(r'

(.)
')# 正则表达式获得特征_1

fearture2_pattern = re.compile(r'

(.*)
')# 正则表达式获得特征_2

name = name_pattern.findall(resp.text) # 提取菜名信息

food = food_pattern.findall(resp.text)# 提取主料信息

fixing = fixing_pattern.findall(resp.text)#提取辅料信息

fearture1 = fearture1_pattern.findall(resp.text) #提取特征_1

fearture2 = fearture2_pattern.findall(resp.text)#提取特征_2

output.write(str(name))#将菜名写入output文件,write函数不能写int类型的参数,所以使用str()转化

output.write('\t')#进入下一个单元格

output.write(str(fearture1))#将特征_1写入output文件

output.write('\t')#进入下一个单元格

output.write(str(fearture2))#将特征_2写入output文件

output.write('\t')#进入下一个单元格

for i in range(len(food)):

for j in range(len(food[i])):

output.write(str(food[i][j])) #写入主料

output.write('\t')

if(len(food)<11):

output.write('\t'*2*(11-len(food))) #每道菜的主料数目不同,该行代码可使表格内容对齐

for i in range(len(fixing)):

for j in range(len(fixing[i])):

output.write(str(fixing[i][j])) #写入辅料

output.write('\t')

output.write('\n') #换行

Step_3 信息导出

def spider():

output = open('E:\\programing\\python\\苏菜_2.xls','w',encoding='utf-8')#创建一个excel文件,编码格式为utf-8

output.write('名称\t做法\t特色\t主料')#写入标题栏

output.write('\t'*22)#使内容对齐

output.write('辅料\n')#写入标题栏

for i in range(len(all_url)):

for j in range(len(all_url[i])):

url2=all_url[i][j]

response = requests.get(url2)#逐个访问网页,获得数据

response.encoding = "utf-8" #设置接收编码格式

get_info(response,output)#处理数据,提取信息

output.close()#关闭文件

主函数

time_start = time.time()#记录程序开始时间

for i in range(1,2):#逐页获取菜谱网页信息

get_all_url(i)

spider()#进行提取处理并导出

duration = 1000#提示音时长,1000毫秒 = 1秒

freq = 440 #提示音频率

time_end=time.time()#记录程序结束时间

print('totally cost',time_end-time_start)#打印程序运行时间

winsound.Beep(freq,duration*10) #响铃提示程序结束

经实验,爬取10000条代码需要用时3453秒左右

最后获得的数据如下

写在后面

我是一个C语言上不了80的小白,全靠某度和某歌东拼西凑我的这个程序,在代码风格与写作等方面存在不可避免地会有一些错误和不足的地方,希望大家多多与我交流,多多批评指教我。

Logo

魔乐社区(Modelers.cn) 是一个中立、公益的人工智能社区,提供人工智能工具、模型、数据的托管、展示与应用协同服务,为人工智能开发及爱好者搭建开放的学习交流平台。社区通过理事会方式运作,由全产业链共同建设、共同运营、共同享有,推动国产AI生态繁荣发展。

更多推荐