摄影:产品经理
薄如蝉翼
我们平时使用Requests的时候,一般是这样写代码的:
import requests
def parse(html):
print('对 html 进行处理')
html = requests.get('url')
parse(html)
这是一种非常常见的直线性思维,我先请求网站拿到 html,然后我再把 html 传给负责处理的函数。在整个过程中,“我“担任着调度的角色。
在这种思维方式的影响下,有些同学即使在使用 aiohttp 写异步爬虫,也是这样写的:
import aiohttp
import asyncio
async def request(url):
async with aiohttp.ClientSession() as session:
resp = await session.get(url)
html = await resp.text(encoding='utf-8')
def parse(html):
print('处理 html')
async def main():
url_list = [url1, url2, url3, url4]
tasks = []
for url in url_list:
tasks.append(request(url))
html_list = await asyncio.gather(*tasks)
for html in html_list:
parse(html)
if __name__ == '__main__':
asyncio.run(main())
确实,这些 URL 的网络请求是异步了,但是却必须等到所有 URL 全部请求完成以后,才能开始处理这些 HTML。假如其中一个 URL 访问只需要1秒钟,其他的 URL 请求需要3秒钟。那么这个1秒钟的请求结束以后,还需要等待2秒,才能开始进行处理。
于是,有些同学会修改代码,多包装一层函数:
import aiohttp
import asyncio
async def request(url):
async with aiohttp.ClientSession() as session:
resp = await session.get(url)
html = await resp.text(encoding='utf-8')
def parse(html):
print('处理 html')
async def get(url):
html = await request(url)
parse(html)
async def main():
url_list = [url1, url2, url3, url4]
tasks = []
for url in url_list:
tasks.append(get(url))
await asyncio.gather(*tasks)
if __name__ == '__main__':
asyncio.run(main())
get() 函数整体负责获取一个 URL 的源代码并对它进行解析。然后让 get() 函数异步。
这样做确实能够解决问题,但是大家如果仔细体会就会发现,在 get() 函数里面的代码写法,还是用的同步处理的思想。
既然要写异步代码,那么我们脑子里就要一直记住——很多个请求会同时发出,但是我们并不知道他们什么时候完成。与其让我们去等待它完成,然后再把完成结果传给另外一个函数。不如让这些请求在结束的时候,自行主动把结果传给处理函数。
有了这种思想以后,我们再来修改一下上面的代码:
import aiohttp
import asyncio
async def request(url, callback):
async with aiohttp.ClientSession() as session:
resp = await session.get(url)
html = await resp.text(encoding='utf-8')
callback(html)
def parse(html):
print('处理 html: ', html)
async def main():
url_list = [
'http://exercise.kingname.info/exercise_middleware_ip/1',
'http://exercise.kingname.info/exercise_middleware_ip/2',
'http://exercise.kingname.info/exercise_middleware_ip/3',
'http://exercise.kingname.info/exercise_middleware_ip/4',
'http://exercise.kingname.info/exercise_middleware_ip/5',
'http://exercise.kingname.info/exercise_middleware_ip/6',
]
tasks = []
for url in url_list:
tasks.append(request(url, parse))
await asyncio.gather(*tasks)
if __name__ == '__main__':
asyncio.run(main())
运行效果如下图所示:
这种写法,初看起来与用 get() 函数包装没什么区别,但是他们在思维方式上却完全不一样。
这种不一样,接下来的几篇文章会进一步演示。
友情提示,五一高能预警。
往期文章
你点的每个“在看”,是对我最大的鼓励
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网
猜你喜欢:- 以案例的形式,来分析用户思维、产品思维和工程思维
- 工程思维与产品思维
- mybatis思维导图,让mybatis不再难懂(二) - java思维导图
- 百度发布智能城市“ACE王牌计划” 李彦宏要用AI思维吊打互联网思维
- 架构师思维
- 架构师思维
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
Python高效开发实战——Django、Tornado、Flask、Twisted(第2版)
刘长龙 / 电子工业出版社 / 2019-1 / 99
也许你听说过全栈工程师,他们善于设计系统架构,精通数据库建模、通用网络协议、后端并发处理、前端界面设计,在学术研究或工程项目上能独当一面。通过对Python 3及相关Web框架的学习和实践,你就可以成为这样的全能型人才。 《Python高效开发实战——Django、Tornado、Flask、Twisted(第2版)》分为3篇:上篇是Python基础,带领初学者实践Python开发环境,掌握......一起来看看 《Python高效开发实战——Django、Tornado、Flask、Twisted(第2版)》 这本书的介绍吧!