最近用 Python 的 asyncio,有好多不懂。。
我想实现的一个需求是这样的,我写一个 http 服务端(正在用 Flask ),接到一个用户浏览器发来的 request 之后,同时向多个上游服务器发送 request ,哪个上游服务器的 response 最快就拿谁的结果,把处理一下响应给用户浏览器。 有点类似 HTTP proxy ,只不过是多个上游服务器。由于需要同时向多个上游服务器发请求,我尝试了用 asyncio+aiohttp 。 可是这用在 Flask 里面应该怎么用啊? 难道我要在每个 route 里面写 loop.run_until_complete 吗?
app = Flask(__name__)
loop = asyncio.get_event_loop()
@app.route("/list1/")
def filelist1():
async def get_file_list1():
await dosomething1()
loop.run_until_complete(get_file_list1())
@app.route("/list2/")
def filelist2():
async def get_file_list2():
await dosomething2()
loop.run_until_complete(get_file_list2())
另外有一个 https://github.com/Hardtack/Flask-aiohttp ,不知道有没有用,不知道这个插件解决了什么问题。。还是说要用tornado?也不知道tornado解决了什么问题。。
同时向多个上游服务器发送请求,收到其中的一个 response 之后,就把其它的 IO cancel 掉。用 asyncio 的好处是可以 cancel 掉 IO 。之前曾经试过用多线程+requests ,但这种做法,当一个请求发出之后,还没收到 response 之前,线程就会被阻塞住,没法取消掉。于是了解了异步 IO ,随时可以 cancel 。 非常感谢 @youyongsong 耐心的分析。
我看完后的理解是, tornado 用一个 loop 来处理所有接收到的 http 请求,如果服务器收到非常频繁而密集的请求,这种模型是很高效的。
而我的使用场景,并不是频繁接收请求,而是频繁发出请求。如果我也用传统方式,每接收到一个请求后, fork 出一个进程 /线程来处理该请求,同时在该进程 /线程中建立一个 loop ,并用这个 loop 来管理频繁发出的请求,是否可行?突然想到,如果所有进程 /线程都共用一个 loop 的话,会产生线程安全问题。 还有一点就是,使用协成,出现异常调试起来麻烦一点 ----------------------- 以下是精选回复-----------------------
答:u need tornado
答:你可以试看看 sanic 。
答:你可以看看 https://zhuanlan.zhihu.com/p/25530067 里面有很多对 asyncio 和 sanic 的内容。强烈关注作者的微信公众号
答:感觉,好像要用异步的话,所有的库都要换成异步了。。还有,异步有异常的话不好调试。
答:从不同的上游服务器拿到的 response 相同吗?还是要再处理后返回给浏览器?
答:这个问题涉及的东西比较多 你要了解:
1. 基于 epoll 等实现的 event loop
2. 然后在此基础上理解类似 nodejs 的基于回调函数的异步处理
3. 然后再了解 python 的 cotoutine 如何基于生成器实现的
4. 然后理解 Python 的 Future 和 js 的 Promise 等如何解决基于回调的异步带来的回调嵌套问题
5. 最后 async await 只是一个生成器协程的语法糖
对比下 gevent, nodejs tornado asyncio golang erlang 和 线程模型,然后你就全都明白了。
答:https://github.com/lbp0200/PRCDNS
大概是需要回调
答:说下我对这 python 这几种 web 模型的理解吧:
首先是 http server + wsgi server(container) + wsgi application 这种传统模型吧:
http server 指的是类似于 nginx 或 apache 的服务
wsgi server 指的是类似 gunicorn 和 uwsgi 这样的服务
wsgi application 指的是 flask django 这样的基于 wsgi 接口的框架运行起来的实例
最初这种模型只是为了方便 web 框架的开发者,不需要每个框架层面都去实现一遍 http server ,就增加了一个 WSGI 中间层协议,框架只要实现这个协议的客户端就可以,然后用 wsgi server 去实现 http 协议的解析并去调用客户端(wsgi application)。
为了方便开发,每个框架都内置了一个简易的 wsgi server ,为什么还要用专门的 wsgi server 呢?
wsgi 除了解析 http 协议以及 http 端口侦听外,还负责了流量转发以及 wsgi application 进程管理的功能。一般 wsgi 框架内置的 wsgi server 都是一个单进程,一次只能处理一个请求。而目的通用的 wsgi server(gunicorn, uwsgi)都至少支持 pre fork 模型,这种模型会起一个 master 来侦听请求,并启动多个 slave(每个 slave 是一个 wsgi application), master 负责把请求转发到空闲的 slave 上。除了这种传统的基于进程的 pre fork 同步模型,不同的 wsgi server 也会支持一些其它模型,有基于线程的同步模型,也有基于 asyncio 的异步模型。
这种模型下怎样写异步代码呢?
1. 直接用传统的异步编程(进程,线程,协程),虽然有些 wsgi server 支持 asynio 模型,但是这也需要用户所写的代码做相应的支持。这就导致了如果我们在 wsgi application 的时候不能随便使用线程和异步 IO ,如果用了就需要配置 wsgi server 使其支持我们自己的写法。因此为了使得我们缩写的 application 能部署在任意的 wsgi server(container)中,我们就只能写同步代码了。
2. 使用分布式异步编程,使用类似 celery 的方式,将需要异步处理的东西发送到 worker 去处理。
既然有了 wsgi server ,为什么还要有一个 http server 呢?
主要是因为 wsgi server 支持的并发量比较低,一般会用一个专门的 http server 来做一层缓冲,避免并发量过大时直接服务挂掉。
python 传统的这种 wsgi 模型,主要是为了方便框架开发者只需要专注框架层面,而非 http 处理层面。但这样却增加了服务部署的复杂度,需要同时部署和配置 http server 和 wsgi server ,如果想支持异步还要部署 worker ,而使用 tornado 或 go 开发的应用因为自己实现了高效 http 处理的应用只需要部署自己就可以了。
接下来是 tornado 和 twisted 这种模型:
这种模型和上面的传统模型处于一个时期,这种模型和 nodejs 差不多,都是基于回调的模型,适用于高 IO 低 CPU 的场景。这种模型自己实现了一个基于回调 http server(event loop),每一个请求都被注册成一个异步函数来处理,然后主循环来不断的循环这些函数。这样就和 pre fork 模型有了区别, pre fork 模型中每一个 slave 都是一个 wsgi application ,一个 wsgi application 都只能处理一个请求,而回调模型只有一个线程,不仅极大的减少了内存的分配还减小了进城以及线程间的切换开销,从而可以支持高 IO 并发。但是这种模型也有很明显的缺点,就是一旦应用程序有大量的 CPU 计算,就会让这个线程堵住,所有的请求都会收到影响,如果应用在处理一个请求时崩溃,所有的请求也都会收到影响。
接下来时 aiohttp/sanic 这种模型:
这种模型和 tornada 模型的改进,但实质上是一样的,因为回调的写法不易读也容易出错,于是将回调的写法改成了同步的写法。这种模型和 koa2 和 go net/http 查不多, asyncio 提供了类似 go coroutine 的功能和写法,而 aiohttp 则提供了类似 go 中的 net/http 的 http 处理库。
答:怎么说呢......Flask 是个 web 框架,但是并不是一个 webserver,这也就是你的前面还要加 nginx/unicorn 的原因(当然一般还会有 uwsgi)......简单但是不准确的来说 flask 只负责处理请求,而不负责收发请求......
而楼上提到的 tornado 有 web 框架,也有 webserver.
另外,楼主这个应该使用负载均衡更好一点吧,这种竞争式的其实一般来说效率反而底下.
答:asyncio 可以看 http://aosabook.org/en/500L/a-web-crawler-with-asyncio-coroutines.html 我这里很早以前糙译了一下 http://drafts.damnever.com/2015/A-Web-Crawler-With-asyncio-Coroutines.html 末尾有惊喜..
答:好贴好回复,mark
0条评论