Python3多线程爬虫实例讲解代码

作者:pythontab 时间:2021-01-10 21:45:28 

多线程概述

多线程使得程序内部可以分出多个线程来做多件事情,充分利用CPU空闲时间,提升处理效率。python提供了两个模块来实现多线程thread 和threading ,thread 有一些缺点,在threading 得到了弥补。并且在Python3中废弃了thread模块,保留了更强大的threading模块。

使用场景

在python的原始解释器CPython中存在着GIL(Global Interpreter Lock,全局解释器锁),因此在解释执行python代码时,会产生互斥锁来限制线程对共享资源的访问,直到解释器遇到I/O操作或者操作次数达到一定数目时才会释放GIL。所以,虽然CPython的线程库直接封装了系统的原生线程,但CPython整体作为一个进程,同一时间只会有一个获得GIL的线程在跑,其他线程则处于等待状态。这就造成了即使在多核CPU中,多线程也只是做着分时切换而已。

如果你的程序是CPU密集型,多个线程的代码很有可能是线性执行的。所以这种情况下多线程是鸡肋,效率可能还不如单线程因为有上下文切换开销。但是如果你的代码是IO密集型,涉及到网络、磁盘IO的任务都是IO密集型任务,多线程可以明显提高效率,例如多线程爬虫,多线程文件处理等等

多线程爬虫

多线程爬虫的代码实例

注: 以下代码在python3下运行通过, python2版本差异较大,不能运行成功,如需帮助请下方留意。


# coding=utf-8
import threading, queue, time, urllib
from urllib import request
baseUrl = 'http://www.pythontab.com/html/pythonjichu/'
urlQueue = queue.Queue()
for i in range(2, 10):
url = baseUrl + str(i) + '.html'
urlQueue.put(url)
#print(url)
def fetchUrl(urlQueue):
while True:
 try:
  #不阻塞的读取队列数据
  url = urlQueue.get_nowait()
  i = urlQueue.qsize()
 except Exception as e:
  break
 print ('Current Thread Name %s, Url: %s ' % (threading.currentThread().name, url))
 try:
  response = urllib.request.urlopen(url)
  responseCode = response.getcode()
 except Exception as e:
  continue
 if responseCode == 200:
  #抓取内容的数据处理可以放到这里
  #为了突出效果, 设置延时
  time.sleep(1)
if __name__ == '__main__':
startTime = time.time()
threads = []
# 可以调节线程数, 进而控制抓取速度
threadNum = 4
for i in range(0, threadNum):
 t = threading.Thread(target=fetchUrl, args=(urlQueue,))
 threads.append(t)
for t in threads:
 t.start()
for t in threads:
 #多线程多join的情况下,依次执行各线程的join方法, 这样可以确保主线程最后退出, 且各个线程间没有阻塞
 t.join()
endTime = time.time()
print ('Done, Time cost: %s ' % (endTime - startTime))

运行结果:

1个线程时:


Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/2.html
Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/3.html
Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/4.html
Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/5.html
Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/6.html
Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/7.html
Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/8.html
Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/9.html
Done, Time cost: 8.182249069213867

2个线程时:


Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/2.html
Current Thread Name Thread-2, Url: http://www.pythontab.com/html/pythonjichu/3.html
Current Thread Name Thread-2, Url: http://www.pythontab.com/html/pythonjichu/4.html
Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/5.html
Current Thread Name Thread-2, Url: http://www.pythontab.com/html/pythonjichu/6.html
Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/7.html
Current Thread Name Thread-2, Url: http://www.pythontab.com/html/pythonjichu/8.html
Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/9.html
Done, Time cost: 4.0987958908081055

3个线程时:


Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/2.html
Current Thread Name Thread-2, Url: http://www.pythontab.com/html/pythonjichu/3.html
Current Thread Name Thread-3, Url: http://www.pythontab.com/html/pythonjichu/4.html
Current Thread Name Thread-4, Url: http://www.pythontab.com/html/pythonjichu/5.html
Current Thread Name Thread-2, Url: http://www.pythontab.com/html/pythonjichu/6.html
Current Thread Name Thread-4, Url: http://www.pythontab.com/html/pythonjichu/7.html
Current Thread Name Thread-1, Url: http://www.pythontab.com/html/pythonjichu/9.html
Current Thread Name Thread-3, Url: http://www.pythontab.com/html/pythonjichu/8.html
Done, Time cost: 2.287320137023926

通过调节线程数可以看到,执行时间会随着线程数的增加而缩短,抓取效率成正比增加。

总结:

Python多线程在IO密集型任务,多线程可以明显提高效率,CPU密集型任务不适合使用多线程处理。

来源:http://www.pythontab.com/html/2018/pythonhexinbiancheng_0103/1215.html

标签:Python,多线程,爬虫
0
投稿

猜你喜欢

  • php链式操作mysql数据库(封装类带使用示例)

    2023-05-25 02:58:22
  • 微信小程序开发之获取用户手机号码(php接口解密)

    2023-11-15 03:34:59
  • 在pytorch 中计算精度、回归率、F1 score等指标的实例

    2022-08-10 06:28:18
  • 举例讲解Python设计模式编程的代理模式与抽象工厂模式

    2021-08-13 06:23:49
  • PHP页面静态化的优缺点与实现

    2023-05-27 09:14:40
  • oracle12C安装步骤(图文详解)

    2023-07-15 14:19:34
  • Python简单进程锁代码实例

    2022-06-23 00:45:22
  • python 多进程通信模块的简单实现

    2021-06-13 08:21:32
  • 网页制作 JSP与ASP 的比较

    2005-08-10 16:00:00
  • Python生成一个迭代器的实操方法

    2022-05-08 09:34:32
  • setInterval 和 setTimeout 会产生内存溢出

    2008-03-08 13:10:00
  • 从MySQL导大量数据的程序实现方法

    2009-03-06 14:34:00
  • 重温Javascript继承机制

    2011-07-04 12:17:23
  • python通过线程实现定时器timer的方法

    2023-04-28 17:49:28
  • python中文分词,使用结巴分词对python进行分词(实例讲解)

    2023-03-15 13:37:30
  • 方便和实用

    2009-02-02 10:16:00
  • python利用datetime模块计算时间差

    2021-10-07 01:02:04
  • 一个css与js结合的下拉菜单支持主流浏览器

    2023-07-18 01:28:20
  • pytorch 实现计算 kl散度 F.kl_div()

    2023-04-03 20:16:18
  • asp中获取内容中所有图片与获取内容中第一个图片的代码

    2011-02-20 10:51:00
  • asp之家 网络编程 m.aspxhome.com