自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Python爬蟲實(shí)戰(zhàn):?jiǎn)尉€程、多線程和協(xié)程性能對(duì)比

開發(fā) 后端
今天我要給大家分享的是如何爬取中農(nóng)網(wǎng)產(chǎn)品報(bào)價(jià)數(shù)據(jù),并分別用普通的單線程、多線程和協(xié)程來爬取,從而對(duì)比單線程、多線程和協(xié)程在網(wǎng)絡(luò)爬蟲中的性能。

[[378975]]

 一、前言

今天我要給大家分享的是如何爬取中農(nóng)網(wǎng)產(chǎn)品報(bào)價(jià)數(shù)據(jù),并分別用普通的單線程、多線程和協(xié)程來爬取,從而對(duì)比單線程、多線程和協(xié)程在網(wǎng)絡(luò)爬蟲中的性能。

目標(biāo)URL:https://www.zhongnongwang.com/quote/product-htm-page-1.html

爬取產(chǎn)品品名、最新報(bào)價(jià)、單位、報(bào)價(jià)數(shù)、報(bào)價(jià)時(shí)間等信息,保存到本地Excel。

二、爬取測(cè)試

翻頁(yè)查看 URL 變化規(guī)律: 

  1. https://www.zhongnongwang.com/quote/product-htm-page-1.html  
  2. https://www.zhongnongwang.com/quote/product-htm-page-2.html  
  3. https://www.zhongnongwang.com/quote/product-htm-page-3.html  
  4. https://www.zhongnongwang.com/quote/product-htm-page-4.html  
  5. https://www.zhongnongwang.com/quote/product-htm-page-5.html  
  6. https://www.zhongnongwang.com/quote/product-htm-page-6.html 

檢查網(wǎng)頁(yè),可以發(fā)現(xiàn)網(wǎng)頁(yè)結(jié)構(gòu)簡(jiǎn)單,容易解析和提取數(shù)據(jù)。

思路:每一條產(chǎn)品報(bào)價(jià)信息在 class 為 tb 的 table 標(biāo)簽下的 tbody 下的 tr 標(biāo)簽里,獲取到所有 tr 標(biāo)簽的內(nèi)容,然后遍歷,從中提取出每一個(gè)產(chǎn)品品名、最新報(bào)價(jià)、單位、報(bào)價(jià)數(shù)、報(bào)價(jià)時(shí)間等信息。 

  1. # -*- coding: UTF-8 -*-  
  2. """  
  3. @File    :demo.py  
  4. @Author  :葉庭云  
  5. @CSDN    :https://yetingyun.blog.csdn.net/  
  6. """  
  7. import requests  
  8. import logging  
  9. from fake_useragent import UserAgent  
  10. from lxml import etree  
  11. # 日志輸出的基本配置  
  12. logging.basicConfig(level=logging.INFO, format='%(asctime)s - %(levelname)s: %(message)s' 
  13. # 隨機(jī)產(chǎn)生請(qǐng)求頭  
  14. ua = UserAgent(verify_ssl=Falsepath='fake_useragent.json' 
  15. url = 'https://www.zhongnongwang.com/quote/product-htm-page-1.html'  
  16. # 偽裝請(qǐng)求頭  
  17. headers = {  
  18.     "Accept-Encoding": "gzip",  # 使用gzip壓縮傳輸數(shù)據(jù)讓訪問更快  
  19.     "User-Agent": ua.random  
  20.  
  21. # 發(fā)送請(qǐng)求  獲取響應(yīng)  
  22. rep = requests.get(url, headersheaders=headers)  
  23. print(rep.status_code)    # 200  
  24. # Xpath定位提取數(shù)據(jù)  
  25. html = etree.HTML(rep.text)  
  26. items = html.xpath('/html/body/div[10]/table/tr[@align="center"]')  
  27. logging.info(f'該頁(yè)有多少條信息:{len(items)}')  # 一頁(yè)有20條信息  
  28. # 遍歷提取出數(shù)據(jù)  
  29. for item in items:  
  30.     name = ''.join(item.xpath('.//td[1]/a/text()'))  # 品名  
  31.     price = ''.join(item.xpath('.//td[3]/text()'))   # 最新報(bào)價(jià)  
  32.     unit = ''.join(item.xpath('.//td[4]/text()'))    # 單位  
  33.     nums = ''.join(item.xpath('.//td[5]/text()'))    # 報(bào)價(jià)數(shù)  
  34.     time_ = ''.join(item.xpath('.//td[6]/text()'))   # 報(bào)價(jià)時(shí)間  
  35.     logging.info([name, price, unit, nums, time_]) 

運(yùn)行結(jié)果如下:

可以成功爬取到數(shù)據(jù),接下來分別用普通的單線程、多線程和協(xié)程來爬取 50 頁(yè)的數(shù)據(jù)、保存到Excel。

三、單線程爬蟲 

  1. # -*- coding: UTF-8 -*-  
  2. """  
  3. @File    :?jiǎn)尉€程.py  
  4. @Author  :葉庭云  
  5. @CSDN    :https://yetingyun.blog.csdn.net/  
  6. """  
  7. import requests  
  8. import logging  
  9. from fake_useragent import UserAgent  
  10. from lxml import etree  
  11. import openpyxl  
  12. from datetime import datetime  
  13. # 日志輸出的基本配置  
  14. logging.basicConfig(level=logging.INFO, format='%(asctime)s - %(levelname)s: %(message)s' 
  15. # 隨機(jī)產(chǎn)生請(qǐng)求頭  
  16. ua = UserAgent(verify_ssl=Falsepath='fake_useragent.json' 
  17. wb = openpyxl.Workbook() 
  18. sheet = wb.active 
  19. sheet.append(['品名', '最新報(bào)價(jià)', '單位', '報(bào)價(jià)數(shù)', '報(bào)價(jià)時(shí)間'])  
  20. start = datetime.now()  
  21. for page in range(1, 51):  
  22.     # 構(gòu)造URL  
  23.     url = f'https://www.zhongnongwang.com/quote/product-htm-page-{page}.html'  
  24.     # 偽裝請(qǐng)求頭 
  25.     headers = {  
  26.         "Accept-Encoding": "gzip",  # 使用gzip壓縮傳輸數(shù)據(jù)讓訪問更快  
  27.         "User-Agent": ua.random  
  28.     }  
  29.     # 發(fā)送請(qǐng)求  獲取響應(yīng)  
  30.     rep = requests.get(url, headersheaders=headers)  
  31.     # print(rep.status_code)  
  32.     # Xpath定位提取數(shù)據(jù)  
  33.     html = etree.HTML(rep.text)  
  34.     items = html.xpath('/html/body/div[10]/table/tr[@align="center"]')  
  35.     logging.info(f'該頁(yè)有多少條信息:{len(items)}')  # 一頁(yè)有20條信息  
  36.     # 遍歷提取出數(shù)據(jù)  
  37.     for item in items:  
  38.         name = ''.join(item.xpath('.//td[1]/a/text()'))  # 品名  
  39.         price = ''.join(item.xpath('.//td[3]/text()'))   # 最新報(bào)價(jià)  
  40.         unit = ''.join(item.xpath('.//td[4]/text()'))    # 單位  
  41.         nums = ''.join(item.xpath('.//td[5]/text()'))    # 報(bào)價(jià)數(shù)  
  42.         time_ = ''.join(item.xpath('.//td[6]/text()'))   # 報(bào)價(jià)時(shí)間  
  43.         sheet.append([name, price, unit, nums, time_])  
  44.         logging.info([name, price, unit, nums, time_])  
  45. wb.save(filename='data1.xlsx' 
  46. delta = (datetime.now() - start).total_seconds()  
  47. logging.info(f'用時(shí):{delta}s') 

運(yùn)行結(jié)果如下:

單線程爬蟲必須上一個(gè)頁(yè)面爬取完成才能繼續(xù)爬取,還可能受當(dāng)時(shí)網(wǎng)絡(luò)狀態(tài)影響,用時(shí)48.528703s,才將數(shù)據(jù)爬取完,速度比較慢。

四、多線程爬蟲 

  1. # -*- coding: UTF-8 -*-  
  2. """  
  3. @File    :多線程.py  
  4. @Author  :葉庭云  
  5. @CSDN    :https://yetingyun.blog.csdn.net/  
  6. """  
  7. import requests  
  8. import logging  
  9. from fake_useragent import UserAgent  
  10. from lxml import etree  
  11. import openpyxl  
  12. from concurrent.futures import ThreadPoolExecutor, wait, ALL_COMPLETED  
  13. from datetime import datetime  
  14. # 日志輸出的基本配置  
  15. logging.basicConfig(level=logging.INFO, format='%(asctime)s - %(levelname)s: %(message)s' 
  16. # 隨機(jī)產(chǎn)生請(qǐng)求頭  
  17. ua = UserAgent(verify_ssl=Falsepath='fake_useragent.json' 
  18. wb = openpyxl.Workbook()  
  19. sheet = wb.active  
  20. sheet.append(['品名', '最新報(bào)價(jià)', '單位', '報(bào)價(jià)數(shù)', '報(bào)價(jià)時(shí)間'])  
  21. start = datetime.now() 
  22. def get_data(page):  
  23.     # 構(gòu)造URL  
  24.     url = f'https://www.zhongnongwang.com/quote/product-htm-page-{page}.html'  
  25.     # 偽裝請(qǐng)求頭  
  26.     headers = {  
  27.         "Accept-Encoding": "gzip",    # 使用gzip壓縮傳輸數(shù)據(jù)讓訪問更快  
  28.         "User-Agent": ua.random  
  29.     }  
  30.     # 發(fā)送請(qǐng)求  獲取響應(yīng)  
  31.     rep = requests.get(url, headersheaders=headers)  
  32.     # print(rep.status_code)  
  33.     # Xpath定位提取數(shù)據(jù)  
  34.     html = etree.HTML(rep.text)  
  35.     items = html.xpath('/html/body/div[10]/table/tr[@align="center"]')  
  36.     logging.info(f'該頁(yè)有多少條信息:{len(items)}')  # 一頁(yè)有20條信息  
  37.     # 遍歷提取出數(shù)據(jù)  
  38.     for item in items:  
  39.         name = ''.join(item.xpath('.//td[1]/a/text()'))   # 品名  
  40.         price = ''.join(item.xpath('.//td[3]/text()'))    # 最新報(bào)價(jià)  
  41.         unit = ''.join(item.xpath('.//td[4]/text()'))     # 單位  
  42.         nums = ''.join(item.xpath('.//td[5]/text()'))     # 報(bào)價(jià)數(shù)  
  43.         time_ = ''.join(item.xpath('.//td[6]/text()'))    # 報(bào)價(jià)時(shí)間  
  44.         sheet.append([name, price, unit, nums, time_])  
  45.         logging.info([name, price, unit, nums, time_]) 
  46. def run():  
  47.     # 爬取1-50頁(yè)  
  48.     with ThreadPoolExecutor(max_workers=6) as executor:  
  49.         future_tasks = [executor.submit(get_data, i) for i in range(1, 51)]  
  50.         wait(future_tasks, return_when=ALL_COMPLETED 
  51.     wb.save(filename='data2.xlsx' 
  52.     delta = (datetime.now() - start).total_seconds()  
  53.     print(f'用時(shí):{delta}s')  
  54. run() 

運(yùn)行結(jié)果如下:

多線程爬蟲爬取效率提升非常可觀,用時(shí) 2.648128s,爬取速度很快。

五、異步協(xié)程爬蟲 

  1. # -*- coding: UTF-8 -*-  
  2. """  
  3. @File    :demo1.py  
  4. @Author  :葉庭云  
  5. @CSDN    :https://yetingyun.blog.csdn.net/  
  6. """  
  7. import aiohttp  
  8. import asyncio  
  9. import logging  
  10. from fake_useragent import UserAgent  
  11. from lxml import etree  
  12. import openpyxl  
  13. from datetime import datetime  
  14. # 日志輸出的基本配置  
  15. logging.basicConfig(level=logging.INFO, format='%(asctime)s - %(levelname)s: %(message)s' 
  16. # 隨機(jī)產(chǎn)生請(qǐng)求頭  
  17. ua = UserAgent(verify_ssl=Falsepath='fake_useragent.json' 
  18. wb = openpyxl.Workbook()  
  19. sheet = wb.active 
  20. sheet.append(['品名', '最新報(bào)價(jià)', '單位', '報(bào)價(jià)數(shù)', '報(bào)價(jià)時(shí)間'])  
  21. start = datetime.now()  
  22. class Spider(object):  
  23.     def __init__(self):  
  24.         # self.semaphore = asyncio.Semaphore(6)  # 信號(hào)量,有時(shí)候需要控制協(xié)程數(shù),防止爬的過快被反爬  
  25.         self.header = {  
  26.                 "Accept-Encoding": "gzip",    # 使用gzip壓縮傳輸數(shù)據(jù)讓訪問更快  
  27.                 "User-Agent": ua.random  
  28.             } 
  29.     async def scrape(self, url):  
  30.         # async with self.semaphore:  # 設(shè)置最大信號(hào)量,有時(shí)候需要控制協(xié)程數(shù),防止爬的過快被反爬  
  31.         session = aiohttp.ClientSession(headers=self.header, connector=aiohttp.TCPConnector(ssl=False))  
  32.         response = await session.get(url)  
  33.         result = await response.text()  
  34.         await session.close()  
  35.         return result  
  36.     async def scrape_index(self, page):  
  37.         url = f'https://www.zhongnongwang.com/quote/product-htm-page-{page}.html'  
  38.         text = await self.scrape(url)  
  39.         await self.parse(text)  
  40.     async def parse(self, text):  
  41.         # Xpath定位提取數(shù)據(jù)  
  42.         html = etree.HTML(text)  
  43.         items = html.xpath('/html/body/div[10]/table/tr[@align="center"]')  
  44.         logging.info(f'該頁(yè)有多少條信息:{len(items)}')  # 一頁(yè)有20條信息  
  45.         # 遍歷提取出數(shù)據(jù)  
  46.         for item in items:  
  47.             name = ''.join(item.xpath('.//td[1]/a/text()'))  # 品名  
  48.             price = ''.join(item.xpath('.//td[3]/text()'))  # 最新報(bào)價(jià)  
  49.             unit = ''.join(item.xpath('.//td[4]/text()'))  # 單位  
  50.             nums = ''.join(item.xpath('.//td[5]/text()'))  # 報(bào)價(jià)數(shù)  
  51.             time_ = ''.join(item.xpath('.//td[6]/text()'))  # 報(bào)價(jià)時(shí)間  
  52.             sheet.append([name, price, unit, nums, time_])  
  53.             logging.info([name, price, unit, nums, time_])  
  54.     def main(self):  
  55.         # 50頁(yè)的數(shù)據(jù)  
  56.         scrape_index_tasks = [asyncio.ensure_future(self.scrape_index(page)) for page in range(1, 51)]  
  57.         loop = asyncio.get_event_loop()  
  58.         tasks = asyncio.gather(*scrape_index_tasks)  
  59.         loop.run_until_complete(tasks)  
  60. if __name__ == '__main__':  
  61.     spider = Spider()  
  62.     spider.main()  
  63.     wb.save('data3.xlsx')  
  64.     delta = (datetime.now() - start).total_seconds()  
  65.     print("用時(shí):{:.3f}s".format(delta)) 

運(yùn)行結(jié)果如下:

而到了協(xié)程異步爬蟲,爬取速度更快,嗖的一下,用時(shí) 0.930s 就爬取完 50 頁(yè)數(shù)據(jù),aiohttp + asyncio 異步爬蟲竟恐怖如斯。異步爬蟲在服務(wù)器能承受高并發(fā)的前提下增加并發(fā)數(shù)量,爬取效率提升是非??捎^的,比多線程還要快一些。

三種爬蟲都將 50 頁(yè)的數(shù)據(jù)爬取下來保存到了本地,結(jié)果如下:

六、總結(jié)回顧

今天我演示了簡(jiǎn)單的單線程爬蟲、多線程爬蟲和協(xié)程異步爬蟲??梢钥吹揭话闱闆r下異步爬蟲速度最快,多線程爬蟲略慢一點(diǎn),單線程爬蟲速度較慢,必須上一個(gè)頁(yè)面爬取完成才能繼續(xù)爬取。

但協(xié)程異步爬蟲相對(duì)來說并不是那么好編寫,數(shù)據(jù)抓取無法使用 request 庫(kù),只能使用aiohttp,而且爬取數(shù)據(jù)量大時(shí),異步爬蟲需要設(shè)置最大信號(hào)量來控制協(xié)程數(shù),防止爬的過快被反爬。所以在實(shí)際編寫 Python 爬蟲時(shí),我們一般都會(huì)使用多線程爬蟲來提速,但必須注意的是網(wǎng)站都有 ip 訪問頻率限制,爬的過快可能會(huì)被封ip,所以一般我們?cè)诙嗑€程提速的同時(shí)可以使用代理 ip 來并發(fā)地爬取數(shù)據(jù)。

  •  多線程(multithreading):是指從軟件或者硬件上實(shí)現(xiàn)多個(gè)線程并發(fā)執(zhí)行的技術(shù)。具有多線程能力的計(jì)算機(jī)因有硬件支持而能夠在同一時(shí)間執(zhí)行多于一個(gè)線程,進(jìn)而提升整體處理性能。具有這種能力的系統(tǒng)包括對(duì)稱多處理機(jī)、多核心處理器以及芯片級(jí)多處理或同時(shí)多線程處理器。在一個(gè)程序中,這些獨(dú)立運(yùn)行的程序片段叫作 "線程" (Thread),利用它編程的概念就叫作 "多線程處理"。
  •  異步(asynchronous):為完成某個(gè)任務(wù),不同程序單元之間過程中無需通信協(xié)調(diào),也能完成任務(wù)的方式,不相關(guān)的程序單元之間可以是異步的。例如,爬蟲下載網(wǎng)頁(yè)。調(diào)度程序調(diào)用下載程序后,即可調(diào)度其他任務(wù),而無需與該下載任務(wù)保持通信以協(xié)調(diào)行為。不同網(wǎng)頁(yè)的下載、保存等操作都是無關(guān)的,也無需相互通知協(xié)調(diào)。這些異步操作的完成時(shí)刻并不確定。簡(jiǎn)言之,異步意味著無序。
  •  協(xié)程(coroutine),又稱微線程、纖程,協(xié)程是一種用戶態(tài)的輕量級(jí)線程。協(xié)程擁有自己的寄存器上下文和棧。協(xié)程調(diào)度切換時(shí),將寄存器上下文和棧保存到其他地方,在切回來的時(shí)候,恢復(fù)先前保存的寄存器上下文和棧。因此協(xié)程能保留上一次調(diào)用時(shí)的狀態(tài),即所有局部狀態(tài)的一個(gè)特定組合,每次過程重入時(shí),就相當(dāng)于進(jìn)入上一次調(diào)用的狀態(tài)。協(xié)程本質(zhì)上是個(gè)單進(jìn)程,協(xié)程相對(duì)于多進(jìn)程來說,無需線程上下文切換的開銷,無需原子操作鎖定及同步的開銷,編程模型也非常簡(jiǎn)單。我們可以使用協(xié)程來實(shí)現(xiàn)異步操作,比如在網(wǎng)絡(luò)爬蟲場(chǎng)景下,我們發(fā)出一個(gè)請(qǐng)求之后,需要等待一定的時(shí)間才能得到響應(yīng),但其實(shí)在這個(gè)等待過程中,程序可以干許多其他的事情,等到響應(yīng)得到之后才切換回來繼續(xù)處理,這樣可以充分利用 CPU 和其他資源,這就是協(xié)程的優(yōu)勢(shì)。 

 

責(zé)任編輯:龐桂玉 來源: Python中文社區(qū) (ID:python-china)
相關(guān)推薦

2020-11-09 09:33:37

多線程

2023-08-17 14:12:17

2021-06-11 11:28:22

多線程fork單線程

2023-12-13 09:56:13

?多進(jìn)程多線程協(xié)程

2024-09-27 11:51:33

Redis多線程單線程

2009-07-10 09:05:20

SwingWorker

2012-02-15 10:26:40

JavaJava Socket

2023-12-01 08:18:24

Redis網(wǎng)絡(luò)

2019-10-29 20:13:43

Java技術(shù)程序員

2020-09-23 13:37:25

Redis6.0

2020-11-17 10:20:53

Redis多線程單線程

2010-08-30 08:55:56

JavaScript引

2022-01-04 11:11:32

Redis單線程Reactor

2018-01-11 08:24:45

服務(wù)器模型詳解

2010-01-28 16:45:44

Android單線程模

2017-03-06 14:08:38

JavaScript單線程setTimeout

2024-02-26 00:00:00

JavaScript單線程高效

2025-04-24 08:15:00

Redis單線程線程

2023-06-08 08:21:08

多線程編程線程間通信

2025-01-17 08:23:33

點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)