在上一篇博客中:python——图片爬虫:爬取爱女神网站(www.znzhi.net)上的妹子图 基础篇
我讲解了图片爬虫的基本步骤,并实现了爬虫代码
在本篇中,我将带领大家对基础篇中的代码进行改善,加入多线程,提高爬取效率。
首先我们明确一个改进的思路,就是在函数downloadAlbum(url)中:
# 循环下载专辑中各个图片
for num in range(1, pic_count+1):
pic_url = url+"/"+str(num)+'.html'
i=0
while downloadOnePic(path, pic_url) == False:
print 'redownload'
i=i+1
if i > 5:
print "timeout's time too much"
break
将for循环体中,调用downloadOnePic()的操作放到子线程中,然后循环创建子线程,即有n个图片,循环创建n个子线程,相当于n个图片同时处于下载状态
一、新建子线程类
# 循环下载专辑中各个图片
for num in range(1, pic_count+1):
pic_url = url+"/"+str(num)+'.html'
i=0
while downloadOnePic(path, pic_url) == False:
print 'redownload'
i=i+1
if i > 5:
print "timeout's time too much"
break
二、循环创建子线程
# 循环下载专辑中各个图片,对每个图片的下载各开启一个线程
for num in range(1, pic_count+1):
pic_url = url+"/"+str(num)+'.html'
# 创建新的子线程
threadD = threadDownload(path,pic_url)
# 开启子线程
threadD.start()
# 在主线程中循环查询当前正在活动的线程数量
while threading.active_count() != 0:
# 当正在活动的线程数量为1,即只剩主线程时,表示所有子线程都已关闭,即所有图片下载完毕
if threading.active_count() == 1:
print ' all pic has downloaded of this page:' + url
return True
在原本for循环体调用downloadOnePic()的地方改为创建新的子线程,并启动
然后用while循环查询当前正在活动的线程数量,并在循环体中判断当前的线程数量,当活动线程数量等于1,即只剩主线程时,表示所有子线程已关闭,所有图片下载完毕,此时return退出
此爬虫已经改进完成,通过使用多线程提高爬取效率。若还有其他可改进的地方,欢迎交流学习。
完整代码如下:
# coding:utf8
# python环境2.7
# 爬取网站:http://www.znzhi.net/
# author:CodeZ
import os
import re
import threading
import urllib2
import time
from bs4 import BeautifulSoup
BASE_PATH = 'picture'
HOST_HOT = 'http://www.znzhi.net/hot'
HOST_ALBUM = 'http://www.znzhi.net/p'
MAX_PAGE_NUM = 387
class threadDownload(threading.Thread):
def __init__(self,path,url):
threading.Thread.__init__(self)
# 路径参数
self.path = path
# url参数
self.url = url
def run(self):
i=0
while downloadOnePic(self.path, self.url) == False:
print 'redownload'
i=i+1
if i > 5:
print "timeout's time too much"
break
def downloadUrl(url):
# 捕获异常(超时)
try:
# 打开网页
response = urllib2.urlopen(url, timeout=10)
# 设置编码方式
response.encoding = 'utf-8'
# 判断http请求的状态
if response.getcode() == 200:
# 状态正常(200),返回页面数据
return response.read()
else:
# 失败,打印消息,返回空数据
print "error:url visit failed"
return ''
except Exception, e:
# 打印异常
print "exception:"+e.message
print "reopen"
# 重新下载
return downloadUrl(url)
# 获取相册名称
def getPicName(picUrl) :
# 截取地址中最后一个/后面的字符,即图片名
picName = os.path.basename(picUrl)
if '.jpg' in picName:
return picName
return 'error.jpg'
# 下载单张照片
def downloadOnePic(path,url):
soup = BeautifulSoup(downloadUrl(url),
'html.parser',
from_encoding='utf-8')
# 获取存有img节点
img_node = soup.find('div', class_='main-image').find('img')
# 获取img的src值,即图片地址
pic_url = img_node.get('src')
# 调用getPicName()获取图片名称
pic_name = getPicName(pic_url).encode('utf-8')
try:
# 访问图片地址,获取数据
content = urllib2.urlopen(pic_url, timeout=10).read()
# 保存图片到本地
with open(path + '/' + pic_name, 'wb') as code:
code.write(content)
print ' -> ' + pic_name + " download success"
#捕获异常
except Exception, e:
print "exception:"+e.message
return False
return True
def downloadAlbum(url):
print "album:"+url
# 获取当前页面数据
content = downloadUrl(url)
# 传入页面数据content,创建beautifulsoup对象soup
soup = BeautifulSoup(content,
'html.parser',
from_encoding='utf-8')
# 获取存有图片专辑标题的h2标签
title = soup.find('div', class_='content').find('h2')
# 检查是否有内容,在实际爬取中,有遇到过空图片专辑的情况,
if title == None:
print "error:web content has lost"
return
# 通过正则筛选出标题中含有的总图片数值
title_num = re.findall(r'\d+', title.get_text())
pic_count = int(title_num[-1])
# 将(1/num)截取去除,并添加总图片数 [num]
title_split = title.get_text().split(' (', 1)
album_title = title_split[0]+'['+str(pic_count)+']'
# 删去标题中的'/'字符,防止在用标题作为名称建图片文件夹时报错
album_title = album_title.replace('/', ' ')
# 拼接本地文件夹路径,并检查路径是否存在,防止重复下载
path = BASE_PATH + "/" + album_title
if os.path.exists(path):
print ' -> ' + album_title + ' has exists'
return True
# 新建存放当前专辑的图片文件夹
checkDocuments(path)
print path
# 新建一个html,存有此图片专辑相关信息
with open(path+'/source.html','w') as fout:
fout.write("<html>")
fout.write("<body>")
fout.write("<p>"+album_title.encode('utf-8')+"-["+str(pic_count)+"p]"+"</p>")
fout.write("<a href=\""+url.encode('utf-8')+"\">来源网址:"+url.encode('utf-8')+"</a>")
fout.write("</body>")
fout.write("</html>")
# 循环下载专辑中各个图片,对每个图片的下载各开启一个线程
for num in range(1, pic_count+1):
pic_url = url+"/"+str(num)+'.html'
# 创建新的子线程
threadD = threadDownload(path,pic_url)
# 开启子线程
threadD.start()
# 在主线程中循环查询当前正在活动的线程数量
while threading.active_count() != 0:
# 当正在活动的线程数量为1,即只剩主线程时,表示所有子线程都已关闭,即所有图片下载完毕
if threading.active_count() == 1:
print ' all pic has downloaded of this page:' + url
return True
def downloadPage(url):
print "page:"+url
# 获取当前页面数据
content = downloadUrl(url)
# 传入页面数据content,创建beautifulsoup对象soup
soup = BeautifulSoup(content,
'html.parser',
from_encoding='utf-8')
# 获取单页中18个图片专辑的父节点
album_block = soup.find('ul', id='images')
# 获取父节点下图片专辑地址的a节点集
album_nodes = album_block.findAll('a', href=re.compile(r'http://www.znzhi.net/p/'))
# 由于每个专辑的a标签有两个,用[::2]获取a节点集中的偶数项,循环下载图片专辑
for album_node in album_nodes[::2]:
# 调用downloadAlbum
# 传入album_node.get('href')获取a节点的href值,即专辑地址
downloadAlbum(album_node.get('href'))
# 若运行中想终止爬虫程序,可在同父目录下新建stop.txt文件
if os.path.exists('stop.txt'):
exit(0)
# 设置图片专辑下载间隙休眠,防止因访问频繁,被网站拉黑
time.sleep(4)
# 检查本地文件路径是否存在,不存在则创建
def checkDocuments(path):
if os.path.exists(path) == False:
os.mkdir(path)
# main函数
if __name__ == "__main__":
# 检查本地下载路径是否存在
checkDocuments(BASE_PATH)
# 循环访问
for i in range(1, MAX_PAGE_NUM+1):
# 拼接页地址,格式为:http://www.znzhi.net/hot/页码.html
page_url = HOST_HOT+'/'+str(i)+'.html'
# 保存当前页码,供查看下载进度
with open('cur_page.txt', 'w') as fpage:
fpage.write(str(i))
# 以页为单位进行下载
downloadPage(page_url)
版权声明:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若内容造成侵权、违法违规、事实不符,请将相关资料发送至xkadmin@xkablog.com进行投诉反馈,一经查实,立即处理!
转载请注明出处,原文链接:https://www.xkablog.com/pythonbc/10316.html