python实现并发爬虫

2024-09-07 17:58
文章标签 python 实现 并发 爬虫

本文主要是介绍python实现并发爬虫,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

阅读目录

一.顺序抓取
二.多线程抓取
三.gevent并发抓取
四.基于tornado的coroutine并发抓取

在进行单个爬虫抓取的时候,我们不可能按照一次抓取一个url的方式进行网页抓取,这样效率低,也浪费了cpu的资源。目前python上面进行并发抓取的实现方式主要有以下几种:进程,线程,协程。进程不在的讨论范围之内,一般来说,进程是用来开启多个spider,比如我们开启了4进程,同时派发4个spider进行网络抓取,每个spider同时抓取4个url。

所以,我们今天讨论的是,在单个爬虫的情况下,尽可能的在同一个时间并发抓取,并且抓取的效率要高。

一.顺序抓取

顺序抓取是最最常见的抓取方式,一般初学爬虫的朋友就是利用这种方式,下面是一个测试代码,顺序抓取8个url,我们可以来测试一下抓取完成需要多少时间:

HEADERS = {'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9',                 'Accept-Language': 'zh-CN,zh;q=0.8',                                                       'Accept-Encoding': 'gzip, deflate',}                                                       
URLS = ['http://www.cnblogs.com/moodlxs/p/3248890.html',                                      'https://www.zhihu.com/topic/19804387/newest',                                        'http://blog.csdn.net/yueguanghaidao/article/details/24281751',                       'https://my.oschina.net/visualgui823/blog/36987',                                     'http://blog.chinaunix.net/uid-9162199-id-4738168.html',                              'http://www.tuicool.com/articles/u67Bz26',                                            'http://rfyiamcool.blog.51cto.com/1030776/1538367/',                                  'http://itindex.net/detail/26512-flask-tornado-gevent']                               #url为随机获取的一批url                                                                               def func():                                                                                   """                                                                                       顺序抓取                                                                                      """                                                                                       import requests                                                                           import time                                                                               urls = URLS                                                                               headers = HEADERS                                                                         headers['user-agent'] = "Mozilla/5.0+(Windows+NT+6.2;+WOW64)+AppleWebKit/537" \           ".36+(KHTML,+like+Gecko)+Chrome/45.0.2454.101+Safari/537.36"      print(u'顺序抓取')                                                                            starttime= time.time()                                                                    for url in urls:                                                                          try:                                                                                  r = requests.get(url, allow_redirects=False, timeout=2.0, headers=headers)        except:                                                                               pass                                                                              else:                                                                                 print(r.status_code, r.url)                                                       endtime=time.time()                                                                       print(endtime-starttime)                                                                  func()

 

我们直接采用内建的time.time()来计时,较为粗略,但可以反映大概的情况。下面是顺序抓取的结果计时:

可以从图片中看到,显示的顺序与urls的顺序是一模一样的,总共耗时为7.763269901275635秒,一共8个url,平均抓取一个大概需要0.97秒。总体来看,还可以接受。

 

二.多线程抓取

线程是python内的一种较为不错的并发方式,我们也给出相应的代码,并且为每个url创建了一个线程,一共8线程并发抓取,下面的代码:

下面是我们运行8线程的测试代码:

 

HEADERS = {'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9',                              'Accept-Language': 'zh-CN,zh;q=0.8',                                                                    'Accept-Encoding': 'gzip, deflate',}                                                                    
URLS = ['http://www.cnblogs.com/moodlxs/p/3248890.html',                                                   'https://www.zhihu.com/topic/19804387/newest',                                                     'http://blog.csdn.net/yueguanghaidao/article/details/24281751',                                    'https://my.oschina.net/visualgui823/blog/36987',                                                  'http://blog.chinaunix.net/uid-9162199-id-4738168.html',                                           'http://www.tuicool.com/articles/u67Bz26',                                                         'http://rfyiamcool.blog.51cto.com/1030776/1538367/',                                               'http://itindex.net/detail/26512-flask-tornado-gevent']                                            def thread():                                                                                              from threading import Thread                                                                           import requests                                                                                        import time                                                                                            urls = URLS                                                                                            headers = HEADERS                                                                                      headers['user-agent'] = "Mozilla/5.0+(Windows+NT+6.2;+WOW64)+AppleWebKit/537.36+" \                    "(KHTML,+like+Gecko)+Chrome/45.0.2454.101+Safari/537.36"                       def get(url):                                                                                          try:                                                                                               r = requests.get(url, allow_redirects=False, timeout=2.0, headers=headers)                     except:                                                                                            pass                                                                                           else:                                                                                              print(r.status_code, r.url)                                                                    print(u'多线程抓取')                                                                                        ts = [Thread(target=get, args=(url,)) for url in urls]                                                 starttime= time.time()                                                                                 for t in ts:                                                                                           t.start()                                                                                          for t in ts:                                                                                           t.join()                                                                                           endtime=time.time()                                                                                    print(endtime-starttime)                                                                               
thread()

多线程抓住的时间如下:

可以看到相较于顺序抓取,8线程的抓取效率明显上升了3倍多,全部完成只消耗了2.154秒。可以看到显示的结果已经不是urls的顺序了,说明每个url各自完成的时间都是不一样的。线程就是在一个进程中不断的切换,让每个线程各自运行一会,这对于网络io来说,性能是非常高的。但是线程之间的切换是挺浪费资源的。

 

三.gevent并发抓取

gevent是一种轻量级的协程,可用它来代替线程,而且,他是在一个线程中运行,机器资源的损耗比线程低很多。如果遇到了网络io阻塞,会马上切换到另一个程序中去运行,不断的轮询,来降低抓取的时间 
下面是测试代码:

HEADERS = {'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9','Accept-Language': 'zh-CN,zh;q=0.8','Accept-Encoding': 'gzip, deflate',}URLS = ['http://www.cnblogs.com/moodlxs/p/3248890.html','https://www.zhihu.com/topic/19804387/newest','http://blog.csdn.net/yueguanghaidao/article/details/24281751','https://my.oschina.net/visualgui823/blog/36987','http://blog.chinaunix.net/uid-9162199-id-4738168.html','http://www.tuicool.com/articles/u67Bz26','http://rfyiamcool.blog.51cto.com/1030776/1538367/','http://itindex.net/detail/26512-flask-tornado-gevent']def main():"""gevent并发抓取"""import requestsimport geventimport timeheaders = HEADERSheaders['user-agent'] = "Mozilla/5.0+(Windows+NT+6.2;+WOW64)+AppleWebKit/537.36+" \"(KHTML,+like+Gecko)+Chrome/45.0.2454.101+Safari/537.36"urls = URLSdef get(url):try:r = requests.get(url, allow_redirects=False, timeout=2.0, headers=headers)except:passelse:print(r.status_code, r.url)print(u'基于gevent的并发抓取')starttime= time.time()g = [gevent.spawn(get, url) for url in urls]gevent.joinall(g)endtime=time.time()print(endtime - starttime)
main()

协程的抓取时间如下:

正常情况下,gevent的并发抓取与多线程的消耗时间差不了多少,但是可能是我网络的原因,或者机器的性能的原因,时间有点长......,请各位小主在自己电脑进行跑一下看运行时间

 

四.基于tornado的coroutine并发抓取

tornado中的coroutine是python中真正意义上的协程,与python3中的asyncio几乎是完全一样的,而且两者之间的future是可以相互转换的,tornado中有与asyncio相兼容的接口。 
下面是利用tornado中的coroutine进行并发抓取的代码:

 

利用coroutine编写并发略显复杂,但这是推荐的写法,如果你使用的是python3,强烈建议你使用coroutine来编写并发抓取。

下面是测试代码:

 

HEADERS = {'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9','Accept-Language': 'zh-CN,zh;q=0.8','Accept-Encoding': 'gzip, deflate',}URLS = ['http://www.cnblogs.com/moodlxs/p/3248890.html','https://www.zhihu.com/topic/19804387/newest','http://blog.csdn.net/yueguanghaidao/article/details/24281751','https://my.oschina.net/visualgui823/blog/36987','http://blog.chinaunix.net/uid-9162199-id-4738168.html','http://www.tuicool.com/articles/u67Bz26','http://rfyiamcool.blog.51cto.com/1030776/1538367/','http://itindex.net/detail/26512-flask-tornado-gevent']
import time
from tornado.gen import coroutine
from tornado.ioloop import IOLoop
from tornado.httpclient import AsyncHTTPClient, HTTPError
from tornado.httpclient import HTTPRequest#urls与前面相同
class MyClass(object):def __init__(self):#AsyncHTTPClient.configure("tornado.curl_httpclient.CurlAsyncHTTPClient")self.http = AsyncHTTPClient()@coroutinedef get(self, url):#tornado会自动在请求首部带上host首部request = HTTPRequest(url=url,method='GET',headers=HEADERS,connect_timeout=2.0,request_timeout=2.0,follow_redirects=False,max_redirects=False,user_agent="Mozilla/5.0+(Windows+NT+6.2;+WOW64)+AppleWebKit/537.36+\(KHTML,+like+Gecko)+Chrome/45.0.2454.101+Safari/537.36",)yield self.http.fetch(request, callback=self.find, raise_error=False)def find(self, response):if response.error:print(response.error)print(response.code, response.effective_url, response.request_time)class Download(object):def __init__(self):self.a = MyClass()self.urls = URLS@coroutinedef d(self):print(u'基于tornado的并发抓取')starttime = time.time()yield [self.a.get(url) for url in self.urls]endtime=time.time()print(endtime-starttime)if __name__ == '__main__':dd = Download()loop = IOLoop.current()loop.run_sync(dd.d)

 

抓取的时间如下:

可以看到总共花费了128087秒,而这所花费的时间恰恰就是最后一个url抓取所需要的时间,tornado中自带了查看每个请求的相应时间。我们可以从图中看到,最后一个url抓取总共花了1.28087秒,相较于其他时间大大的增加,这也是导致我们消耗时间过长的原因。那可以推断出,前面的并发抓取,也在这个url上花费了较多的时间。

总结: 
以上测试其实非常的不严谨,因为我们选取的url的数量太少了,完全不能反映每一种抓取方式的优劣。如果有一万个不同的url同时抓取,那么记下总抓取时间,是可以得出一个较为客观的结果的。 
并且,已经有人测试过,多线程抓取的效率是远不如gevent的。所以,如果你使用的是python2,那么我推荐你使用gevent进行并发抓取;如果你使用的是python3,我推荐你使用tornado的http客户端结合coroutine进行并发抓取。从上面的结果来看,tornado的coroutine是高于gevent的轻量级的协程的。但具体结果怎样,我没测试过。

这篇关于python实现并发爬虫的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/1145757

相关文章

golang 对象池sync.Pool的实现

《golang对象池sync.Pool的实现》:本文主要介绍golang对象池sync.Pool的实现,用于缓存和复用临时对象,以减少内存分配和垃圾回收的压力,下面就来介绍一下,感兴趣的可以了解... 目录sync.Pool的用法原理sync.Pool 的使用示例sync.Pool 的使用场景注意sync.

IDEA实现回退提交的git代码(四种常见场景)

《IDEA实现回退提交的git代码(四种常见场景)》:本文主要介绍IDEA实现回退提交的git代码(四种常见场景),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1.已提交commit,还未push到远端(Undo Commit)2.已提交commit并push到

Kotlin Compose Button 实现长按监听并实现动画效果(完整代码)

《KotlinComposeButton实现长按监听并实现动画效果(完整代码)》想要实现长按按钮开始录音,松开发送的功能,因此为了实现这些功能就需要自己写一个Button来解决问题,下面小编给大... 目录Button 实现原理1. Surface 的作用(关键)2. InteractionSource3.

java对接第三方接口的三种实现方式

《java对接第三方接口的三种实现方式》:本文主要介绍java对接第三方接口的三种实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录HttpURLConnection调用方法CloseableHttpClient调用RestTemplate调用总结在日常工作

golang中slice扩容的具体实现

《golang中slice扩容的具体实现》Go语言中的切片扩容机制是Go运行时的一个关键部分,它确保切片在动态增加元素时能够高效地管理内存,本文主要介绍了golang中slice扩容的具体实现,感兴趣... 目录1. 切片扩容的触发append 函数的实现2. runtime.growslice 函数gro

python进行while遍历的常见错误解析

《python进行while遍历的常见错误解析》在Python中选择合适的遍历方式需要综合考虑可读性、性能和具体需求,本文就来和大家讲解一下python中while遍历常见错误以及所有遍历方法的优缺点... 目录一、超出数组范围问题分析错误复现解决方法关键区别二、continue使用问题分析正确写法关键点三

golang实现动态路由的项目实践

《golang实现动态路由的项目实践》本文主要介绍了golang实现动态路由项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习... 目录一、动态路由1.结构体(数据库的定义)2.预加载preload3.添加关联的方法一、动态路由1

使用Python实现调用API获取图片存储到本地的方法

《使用Python实现调用API获取图片存储到本地的方法》开发一个自动化工具,用于从JSON数据源中提取图像ID,通过调用指定API获取未经压缩的原始图像文件,并确保下载结果与Postman等工具直接... 目录使用python实现调用API获取图片存储到本地1、项目概述2、核心功能3、环境准备4、代码实现

MySQL数据库实现批量表分区完整示例

《MySQL数据库实现批量表分区完整示例》通俗地讲表分区是将一大表,根据条件分割成若干个小表,:本文主要介绍MySQL数据库实现批量表分区的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考... 目录一、表分区条件二、常规表和分区表的区别三、表分区的创建四、将既有表转换分区表脚本五、批量转换表为分区

8种快速易用的Python Matplotlib数据可视化方法汇总(附源码)

《8种快速易用的PythonMatplotlib数据可视化方法汇总(附源码)》你是否曾经面对一堆复杂的数据,却不知道如何让它们变得直观易懂?别慌,Python的Matplotlib库是你数据可视化的... 目录引言1. 折线图(Line Plot)——趋势分析2. 柱状图(Bar Chart)——对比分析3