Python中Scrapy爬虫图片处理详解


Posted in Python onNovember 29, 2017

下载图片

下载图片有两种方式,一种是通过 Requests 模块发送 get 请求下载,另一种是使用 Scrapy 的 ImagesPipeline 图片管道类,这里主要讲后者。

安装 Scrapy 时并没有安装图像处理依赖包 Pillow,需手动安装否则运行爬虫出错。

首先在 settings.py 中设置图片的存储路径:

IMAGES_STORE = 'D:/'

图片处理相关的选项还有:

# 图片最小高度和宽度设置,可以过滤太小的图片
IMAGES_MIN_HEIGHT = 110
IMAGES_MIN_WIDTH = 110

# 生成缩略图选项
IMAGES_THUMBS = {
 'small': (50, 50),
 'big': (270, 270),
}

之前已经存在提取内容的 TuchongPipeline 类,如果使用 ImagePipeline 可以将提取内容的操作都合并过来,但是为了更好的说明图片管道的作用,我们再单独创建一个 ImagePipeline 类,加到 pipelines.py 文件中,同时重载函数 get_media_requests:

class PhotoGalleryPipeline(object):
 ...

class PhotoPipeline(ImagesPipeline):
 def get_media_requests(self, item, info):
  for (id, url) in item['images'].items():
   yield scrapy.Request(url)

上篇文章中我们把图片的URL保存在了 item['images'] 中,它是一个字典类型的数组,形如:[{img_id: img_url}, ...],此函数中需要把 img_url 取出并构建为 scrapy.Request 请求对象并返回,每一个请求都将触发一次下载图片的操作。

到 settings.py 中注册 PhotoPipeline,并把优先级设的比提取内容的管道要高一些,保证图片下载优先于内容处理,目的是如果有图片下载未成功,通过触发 DropItem 异常可以中断这一个 Item 的处理,防止不完整的数据进入下一管道:

ITEM_PIPELINES = {
 'Toutiao.pipelines.PhotoGalleryPipeline': 300,
 'Toutiao.pipelines.PhotoPipeline': 200,
}

执行爬虫 scrapy crawl photo ,如无错误,在设定的存储目录中会出现一个 full 目录,里面是下载后的图片。

文件名处理

下载的文件名是以图片URL通过 sha1 编码得到的字符,类似 0a79c461a4062ac383dc4fade7bc09f1384a3910.jpg 不是太友好,可以通过重载 file_path 函数自定义文件名,比如可以这样保留原文件名:

...
 def file_path(self, request, response=None, info=None):
  file_name = request.url.split('/')[-1]
  return 'full/%s' % (file_name)
...

上面这样处理难免会有重名的文件被覆盖,但参数 request 中没有过多的信息,不便于对图片分类,因此可以改为重载 item_completed 函数,在下载完成后对图片进行分类操作。

函数 item_completed 的定义:

def item_completed(self, results, item, info)

参数中包含 item ,有我们抓取的所有信息,参数 results 为下载图片的结果数组,包含下载后的路径以及是否成功下载,内容如下:

[(True,
 {'checksum': '2b00042f7481c7b056c4b410d28f33cf',
 'path': 'full/0a79c461a4062ac383dc4fade7bc09f1384a3910.jpg',
 'url': 'http://www.example.com/files/product1.pdf'}),
 (False,
 Failure(...))]

重载该函数将下载图片转移到分类目录中,同时关联文件路径到 item 中,保持内容与图片为一个整体:

def item_completed(self, results, item, info):
 image_paths = {x['url'].split('/')[-1]: x['path'] for ok, x in results if ok}
 if not image_paths:
  # 下载失败忽略该 Item 的后续处理
  raise DropItem("Item contains no files")
 else:
  # 将图片转移至以 post_id 为名的子目录中
  for (dest, src) in image_paths.items():
   dir = settings.IMAGES_STORE
   newdir = dir + os.path.dirname(src) + '/' + item['post_id'] + '/'
   if not os.path.exists(newdir):
    os.makedirs(newdir)
   os.rename(dir + src, newdir + dest)
 # 将保存路径保存于 item 中(image_paths 需要在 items.py 中定义)
 item['image_paths'] = image_paths
 return item

接下来在原 TuchongPipeline 类中写入数据库的操作中,通过 item['image_paths'] 路径信息写入本地图片链接。

除了 ImagesPipeline 处理图片外,还有 FilesPipeline 可以处理文件,使用方法与图片类似,事实上 ImagesPipeline 是 FilesPipeline 的子类,因为图片也是文件的一种。

Python 相关文章推荐
Python2.5/2.6实用教程 入门基础篇
Nov 29 Python
在Python的Django框架中包装视图函数
Jul 20 Python
python3+PyQt5实现支持多线程的页面索引器应用程序
Apr 20 Python
数据清洗--DataFrame中的空值处理方法
Jul 03 Python
使用python获取电脑的磁盘信息方法
Nov 01 Python
Python在图片中插入大量文字并且自动换行
Jan 02 Python
Python 中的 global 标识对变量作用域的影响
Aug 12 Python
Python:slice与indices的用法
Nov 25 Python
手动安装python3.6的操作过程详解
Jan 13 Python
spyder 在控制台(console)执行python文件,debug python程序方式
Apr 20 Python
python的链表基础知识点
Sep 13 Python
python 使用paramiko模块进行封装,远程操作linux主机的示例代码
Dec 03 Python
Python使用django框架实现多人在线匿名聊天的小程序
Nov 29 #Python
Python实现的计数排序算法示例
Nov 29 #Python
Scrapy框架CrawlSpiders的介绍以及使用详解
Nov 29 #Python
pycharm下打开、执行并调试scrapy爬虫程序的方法
Nov 29 #Python
Python快速排序算法实例分析
Nov 29 #Python
Python3学习urllib的使用方法示例
Nov 29 #Python
Python实现的选择排序算法示例
Nov 29 #Python
You might like
php zend 相对路径问题
2009/01/12 PHP
PHP中include与require使用方法区别详解
2013/10/19 PHP
laravel5.4生成验证码的实例讲解
2017/08/05 PHP
PHP htmlspecialchars_decode()函数用法讲解
2019/03/01 PHP
laravel实现一个上传图片的接口,并建立软链接,访问图片的方法
2019/10/12 PHP
漂亮的仿flash菜单,来自蓝色经典
2006/06/26 Javascript
JS控制显示隐藏兼容问题(IE6、IE7、IE8)
2010/04/01 Javascript
jquery validate使用攻略 第四步
2010/07/01 Javascript
用jquery实现的模拟QQ邮箱里的收件人选取及其他效果(一)
2011/01/06 Javascript
JavaScript定义类的几种方式总结
2014/01/06 Javascript
IE浏览器中图片onload事件无效的解决方法
2014/04/29 Javascript
JS实现仿Windows7风格的网页右键菜单效果代码
2015/09/11 Javascript
chrome浏览器如何断点调试异步加载的JS
2016/09/05 Javascript
python django事务transaction源码分析详解
2017/03/17 Python
Django之模型层多表操作的实现
2019/01/08 Python
Python如何调用外部系统命令
2019/08/07 Python
Python如何截图保存的三种方法(小结)
2020/09/01 Python
15个应该掌握的Jupyter Notebook使用技巧(小结)
2020/09/23 Python
10个很棒的 CSS3 开发工具 推荐
2011/05/16 HTML / CSS
HTML5 Canvas旋转动画的2个代码例子(一个旋转的太极图效果)
2014/04/10 HTML / CSS
HTML5 video 事件应用示例
2014/09/11 HTML / CSS
HTML最新标准HTML5总结(必看)
2016/06/13 HTML / CSS
Kathmandu澳洲户外商店:新西兰户外运动品牌
2017/11/12 全球购物
最好的意大利皮夹克:D’Arienzo
2018/12/04 全球购物
成人大专生实习期的自我评价
2013/10/02 职场文书
试用期员工考核制度
2014/01/22 职场文书
县优秀教师事迹材料
2014/01/31 职场文书
档案室主任岗位职责
2014/02/12 职场文书
《草原》教学反思
2014/02/15 职场文书
文科毕业生自荐书范文
2014/04/17 职场文书
体育系毕业生自荐信
2014/06/28 职场文书
竞选班干部演讲稿400字
2014/08/20 职场文书
民主评议党员登记表自我评价
2014/10/20 职场文书
2014年班组建设工作总结
2014/12/01 职场文书
在酒桌上的敬酒词
2015/08/12 职场文书
阿里云服务器Ubuntu 20.04上安装Odoo 15
2022/05/20 Servers