在学习协程的时候,会有一个疑问,使用协程语法进行异步请求时,比如async + requests,会有用吗?
其实细想一下就知道,由于requests库是用同步的方式写的,因此async + requests是肯定没用的。
但是本着实践出真知的思想,顺便复习巩固一下多线程、async、aiohttp的写法,还是手动来验证一下。
为了规避网络波动等影响,在本地用Flask搭建一个简易的服务器用于测试。
文章结构如下:
文章目录
先放结论:
threading + requests 能够并发请求async + requests 不能并发请求async + aiohttp 能并发请求因此在进行爬虫的时候,要想加快效率,要么使用threading + requests ,要么就使用async + aiohttp
安装测试所需要的库
pip install flask
pip install requets
pip install aiohttp
在任意路径创建一个文件夹(文件夹名随意),例如./async_test
在该文件夹下创建一个空的py文件app.py用于后续搭建测试用后端。
再创建3个py文件分别对应3个实验,创建完毕后文件目录结构如下(此时的py文件都是空的)
|- async_test
|- app.py
|- 1_threading_requests.py
|- 2_async_requests.py
|- 3_async_aiohttp.py
让每次请求的时候先沉睡2秒,再返回结果,以此来模拟网络延迟。
在app.py文件中添加如下代码
## app.py ##
from flask import Flask
import time
app = Flask(__name__)
@app.route("/")
def index():
time.sleep(2)
return "Hello World!"
if __name__ == '__main__':
app.run()
在./async_test目录下运行
python app.py
* Serving Flask app "app" (lazy loading)
* Environment: production
WARNING: This is a development server. Do not use it in a production deployment.
Use a production WSGI server instead.
* Debug mode: off
* Running on http://127.0.0.1:5000/ (Press CTRL+C to quit)
访问 http://127.0.0.1:5000/ 延迟2秒后会看到Hello World!
完成这一步就搭建好了测试用后端
在1_threading_requests.py文件中添加如下代码
## 1_threading_requests.py ##
import time
import threading
import requests
def get(i):
print(time.strftime('%X'), 'start', i)
resp = requests.get('http://127.0.0.1:5000/')
print(time.strftime('%X'), 'end', i)
start = time.perf_counter()
for i in range(4):
threading.Thread(target=get, args=(i,)).start()
print(f'total {time.perf_counter() - start:.2f}s ')
在./async_test目录下运行
python 1_threading_requests.py
09:23:19 start 0
09:23:19 start 1
09:23:19 start 2
09:23:19 start 3
09:23:21 end 2
09:23:21 end 0
09:23:21 end 3
09:23:21 end 1
发现使用多线程的写法是能够并发请求的。
在2_async_requests.py文件中添加如下代码
## 2_async_requests.py ##
import time
import asyncio
import requests
async def get(i):
print(time.strftime('%X'), 'start', i)
resp = requests.get('http://127.0.0.1:5000/')
print(time.strftime('%X'), 'end', i)
async def main():
for i in range(4):
asyncio.create_task(get(i))
asyncio.run(main())
在./async_test目录下运行
python 2_async_requests.py
09:27:11 start 0
09:27:13 end 0
09:27:13 start 1
09:27:15 end 1
09:27:15 start 2
09:27:17 end 2
09:27:17 start 3
09:27:19 end 3
发现async+requests的写法,代码是顺序执行的,异步并没有起到效果
于是将get(i)函数用aiohttp重写
在3_async_aiohttp.py文件中添加如下代码
## 3_async_aiohttp.py ##
import time
import asyncio
import aiohttp
import requests
async def get(i):
print(time.strftime('%X'), 'start', i)
async with aiohttp.ClientSession() as session:
async with session.get('http://127.0.0.1:5000/') as response:
html = await response.text()
print(time.strftime('%X'), 'end', i)
async def main():
tasks = [asyncio.create_task(get(i)) for i in range(4)]
await asyncio.gather(*tasks)
asyncio.run(main())
在./async_test目录下运行
python 3_async_aiohttp.py
09:37:43 start 0
09:37:43 start 1
09:37:43 start 2
09:37:43 start 3
09:37:45 end 0
09:37:45 end 2
09:37:45 end 3
09:37:45 end 1
发现代码成功异步执行了,总耗时只有两秒
说明python的协程语法需要配合异步python库才会生效。
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
在我的Controller中,我通过以下方式在我的index方法中支持HTML和JSON:respond_todo|format|format.htmlformat.json{renderjson:@user}end在浏览器中拉起它时,它会自然地以HTML呈现。但是,当我对/user资源进行内容类型为application/json的curl调用时(因为它是索引方法),我仍然将HTML作为响应。如何获取JSON作为响应?我还需要说明什么? 最佳答案 您应该将.json附加到请求的url,提供的格式在routes.rb的路径中定义。这
我想在Ruby中创建一个用于开发目的的极其简单的Web服务器(不,不想使用现成的解决方案)。代码如下:#!/usr/bin/rubyrequire'socket'server=TCPServer.new('127.0.0.1',8080)whileconnection=server.acceptheaders=[]length=0whileline=connection.getsheaders想法是从命令行运行这个脚本,提供另一个脚本,它将在其标准输入上获取请求,并在其标准输出上返回完整的响应。到目前为止一切顺利,但事实证明这真的很脆弱,因为它在第二个请求上中断并出现错误:/usr/b
rails中是否有任何规定允许站点的所有AJAXPOST请求在没有authenticity_token的情况下通过?我有一个调用Controller方法的JqueryPOSTajax调用,但我没有在其中放置任何真实性代码,但调用成功。我的ApplicationController确实有'request_forgery_protection'并且我已经改变了config.action_controller.consider_all_requests_local在我的environments/development.rb中为false我还搜索了我的代码以确保我没有重载ajaxSend来发送
这个问题在这里已经有了答案:关闭10年前。PossibleDuplicate:Pythonconditionalassignmentoperator对于这样一个简单的问题表示歉意,但是谷歌搜索||=并不是很有帮助;)Python中是否有与Ruby和Perl中的||=语句等效的语句?例如:foo="hey"foo||="what"#assignfooifit'sundefined#fooisstill"hey"bar||="yeah"#baris"yeah"另外,类似这样的东西的通用术语是什么?条件分配是我的第一个猜测,但Wikipediapage跟我想的不太一样。
什么是ruby的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht
华为OD机试题本篇题目:明明的随机数题目输入描述输出描述:示例1输入输出说明代码编写思路最近更新的博客华为od2023|什么是华为od,od薪资待遇,od机试题清单华为OD机试真题大全,用Python解华为机试题|机试宝典【华为OD机试】全流程解析+经验分享,题型分享,防作弊指南华为o
我想解析一个已经存在的.mid文件,改变它的乐器,例如从“acousticgrandpiano”到“violin”,然后将它保存回去或作为另一个.mid文件。根据我在文档中看到的内容,该乐器通过program_change或patch_change指令进行了更改,但我找不到任何在已经存在的MIDI文件中执行此操作的库.他们似乎都只支持从头开始创建的MIDI文件。 最佳答案 MIDIpackage会为您完成此操作,但具体方法取决于midi文件的原始内容。一个MIDI文件由一个或多个音轨组成,每个音轨是十六个channel中任何一个上的
1.错误信息:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:requestcanceledwhilewaitingforconnection(Client.Timeoutexceededwhileawaitingheaders)或者:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:TLShandshaketimeout2.报错原因:docker使用的镜像网址默认为国外,下载容易超时,需要修改成国内镜像地址(首先阿里