利用tornado使请求实现异步非阻塞
基本IO模型
网上搜了很多关于同步异步,阻塞非阻塞的说法,理解还是不能很透彻,有必要买书看下。
参考:使用异步 I/O 大大提高应用程序的性能
怎样理解阻塞非阻塞与同步异步的区别?
同步和异步:主要关注消息通信机制(重点在B?)。
同步:A调用B,B处理直到获得结果,才返回给A。
异步:A调用B,B直接返回。无需等待结果,B通过状态,通知等来通知A或回调函数来处理。阻塞和非阻塞:主要关注程序等待(重点在A?)。
阻塞:A调用B,A被挂起直到B返回结果给A,A继续执行。
非阻塞:A调用B,A不会被挂起,A可以执行其他操作(但可能A需要轮询检查B是否返回)。同步阻塞:A调用B,A挂起,B处理直到获得结果,返回给A,A继续执行。
同步非阻塞:A调用B,A继续执行,B处理直到获得结果,处理的同时A轮询检查B是否返回结果。
异步阻塞:异步阻塞 I/O 模型的典型流程 (select)。
异步非阻塞:A调用B,B立即返回,A继续执行,B得到结果后通过状态,通知等通知A或回调函数处理。
tornado实现异步非阻塞
参考:
利用异步方法(回调)和@tornado.web.asynchronous
@tornado.web.asynchronous
并不能将一个同步方法变成异步,所以修饰在同步方法上是无效的,只是告诉框架,这个方法是异步的,且只能适用于HTTP verb方法(get、post、delete、put等)。@tornado.web.asynchronous
装饰器适用于callback-style的异步方法,如果是协程则可以用@tornado.gen.coroutine
来修饰。对于用@tornado.web.asynchronous
修饰的异步方法,需要主动self.finish()
来结束该请求,普通的方法(get()
等)会自动结束请求在方法返回的时候。
最基本的使用callback-style的例子,直接使用异步方法,并定义callback方法。
# sync blocking
class SleepHandler(BaseHandler):
# no effective
@tornado.web.asynchronous
def get(self):
time.sleep(5)
self.write('sleep for 5s')
class SleepHandler(BaseHandler):
@tornado.web.asynchronous
def get(self):
tornado.ioloop.IOLoop.instance().add_timeout(time.time() + 5, callback=self.on_response)
def on_response(self):
self.write('sleep for 5s')
self.finish()
# call back
class MyRequestHandler(BaseHandler):
@tornado.web.asynchronous
def get(self):
http = httpclient.AsyncHTTPClient()
http.fetch('http://www.baidu.com', self._on_download)
def _on_download(self, response):
self.write(response.body)
self.finish()
利用ThreadPoolExecutor
利用ThreadPoolExecutor的submit和future对象的add_done_callback
方法,将一个同步方法变成异步。如下例所示,若没有无参可以不用partial()
:
class SleepHandler(BaseHandler):
@tornado.web.asynchronous
def get(self):
sleep_time = 5
def callback(future):
self.write(future.result())
self.finish()
EXECUTOR.submit(partial(self.get_sleep, sleep_time)).add_done_callback(
lambda future: tornado.ioloop.IOLoop.instance().add_callback(
partial(callback, future)))
def get_sleep(self, sleep_time):
time.sleep(sleep_time)
return "Awake! %s" % time.time()
分解一下:
future = EXECUTOR.submit(partial(self.get_sleep, sleep_time))
返回了一个future
对象。future.add_done_callback()
给future
添加一个完成回调函数callback_func
。- 实际上这个回调函数是
callback_func = lambda future: tornado.ioloop.IOLoop.instance().add_callback(partial(callback, future))```
4. 结合起来就是:
class SleepHandler(BaseHandler):
@tornado.web.asynchronous
def get(self):
sleep_time = 5
def final_callback(future_param):
self.write(future_param.result())
self.finish()
def executor_callback(future_param):
tornado.ioloop.IOLoop.instance().add_callback(partial(final_callback, future_param))
future = EXECUTOR.submit(partial(self.get_sleep, sleep_time))
future.add_done_callback(executor_callback)
def get_sleep(self, sleep_time):
time.sleep(sleep_time)
return "Awake! %s" % time.time()
5. 没看文档,源码前本认为为什么要多此一举进行两次callback,直接
tornado.ioloop.IOLoop.instance().add_callback(partial(final_callback, future))
不就ok了吗,为何还要再加上一层
future.add_done_callback(executor_callback)
但发现直接这样是会阻塞IOLoop的。查阅相关文档后发现,```
tornado.ioloop.IOLoop.instance().add_callback
``` 这个方法会将控制权从其他线程转到IOLoop线程上,直接用
tornado.ioloop.IOLoop.instance().add_callback(partial(final_callback, future))
时,
final_callback()中获取
future_param.result()仍然会阻塞,所以需要
future.add_done_callback()```,在该线程完成获取结果后在回callback给IOLoop。
另一种类似的写法,将上例的方法写成了一个装饰器,见下例,但个人认为利用这种方式异步时没有必要单独写一个装饰器吧,而且也不通用吧?
def unblock(f):
@tornado.web.asynchronous
@wraps(f)
def wrapper(*args, **kwargs):
self = args[0]
def callback(future):
self.write(future.result())
self.finish()
EXECUTOR.submit(
partial(f, *args, **kwargs)
).add_done_callback(
lambda future: tornado.ioloop.IOLoop.instance().add_callback(
partial(callback, future)))
return wrapper
class SleepHandler(BaseHandler):
@unblock
def get(self):
time.sleep(5)
return "Awake! %s" % time.time()
使用tornado.concurrent.run_on_executor
简化
上述的两例都相当于开启了新的线程池?
除此之外还可以利用@run_on_executor
装饰器将同步阻塞函数变成异步(或者说被tornado的装饰器理解和识别)。首先阅读一下@run_on_executor
源码:
def run_on_executor(*args, **kwargs):
"""Decorator to run a synchronous method asynchronously on an executor.
The decorated method may be called with a ``callback`` keyword
argument and returns a future.
The `.IOLoop` and executor to be used are determined by the ``io_loop``
and ``executor`` attributes of ``self``. To use different attributes,
pass keyword arguments to the decorator::
@run_on_executor(executor='_thread_pool')
def foo(self):
pass
.. versionchanged:: 4.2
Added keyword arguments to use alternative attributes.
"""
def run_on_executor_decorator(fn):
executor = kwargs.get("executor", "executor")
io_loop = kwargs.get("io_loop", "io_loop")
@functools.wraps(fn)
def wrapper(self, *args, **kwargs):
callback = kwargs.pop("callback", None)
future = getattr(self, executor).submit(fn, self, *args, **kwargs)
if callback:
getattr(self, io_loop).add_future(
future, lambda future: callback(future.result()))
return future
return wrapper
if args and kwargs:
raise ValueError("cannot combine positional and keyword args")
if len(args) == 1:
return run_on_executor_decorator(args[0])
elif len(args) != 0:
raise ValueError("expected 1 argument, got %d", len(args))
return run_on_executor_decorator
可以很快发现在不存在callback
关键字参数时,该装饰器返回了一个future对象,由此并结合上述两例子可以很快的利用@run_on_executor
写出,感觉相当于简化了上述两例的代码并使之变得通用:
class SleepHandler(BaseHandler):
executor = ThreadPoolExecutor(2)
@tornado.web.asynchronous
def get(self):
def callback(future_param):
self.write('sleep %ss' % future_param.result())
self.finish()
future = self.sleep()
future.add_done_callback(lambda f: tornado.ioloop.IOLoop.instance().add_callback(
partial(callback, f)))
@run_on_executor
def sleep(self):
time.sleep(5)
return 5
当然也可以利用callback参数:
class SleepHandler(BaseHandler):
executor = ThreadPoolExecutor(2)
io_loop = tornado.ioloop.IOLoop.instance()
@tornado.web.asynchronous
def get(self):
def callback(res):
self.write('sleep %ss' % res)
self.finish()
self.sleep(callback=callback)
@run_on_executor
def sleep(self):
time.sleep(5)
return 5
协程方式:@tornado.gen.coroutine
和yield
除了上述的用利用@tornado.web.asynchronous
和callback的方式,还可以用@tornado.gen.coroutine
和yield
,如下例子:
class GenRequestHandler(BaseHandler):
@tornado.gen.coroutine
def get(self):
http = httpclient.AsyncHTTPClient()
res = yield http.fetch('http://www.baidu.com')
self.write(res.body)
参考官方文档
Most asynchronous functions in Tornado return a Future; yielding this object returns its result.
利用tornado.gen.Task
修饰一个callback型的异步函数使其能够与yield
使用。
gen.Task is now a function that returns a Future
看一下例子(对于sleep可以直接用yield tornado.gen.sleep(5)
):
# gen.Task
class SleepHandler(BaseHandler):
@tornado.gen.coroutine
def get(self):
yield tornado.gen.Task(tornado.ioloop.IOLoop.instance().add_timeout, time.time() + 5)
# yield tornado.gen.sleep(5)
self.write('sleep for 5s')
还可以用结合celery使用(但并不是最好的方案)。
Last
https://github.com/tornadoweb/tornado/wiki/Links tornado的wiki上有很多异步相关支持的库。
作者:蒋狗
链接:https://www.jianshu.com/p/ef01c1386933
來源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。
利用tornado使请求实现异步非阻塞的更多相关文章
- Python web框架 Tornado(二)异步非阻塞
异步非阻塞 阻塞式:(适用于所有框架,Django,Flask,Tornado,Bottle) 一个请求到来未处理完成,后续一直等待 解决方案:多线程,多进程 异步非阻塞(存在IO请求): Torna ...
- 使用tornado让你的请求异步非阻塞
http://www.dongwm.com/archives/shi-yong-tornadorang-ni-de-qing-qiu-yi-bu-fei-zu-sai/?utm_source=tuic ...
- 在nginx启动后,如果我们要操作nginx,要怎么做呢 别增加无谓的上下文切换 异步非阻塞的方式来处理请求 worker的个数为cpu的核数 红黑树
nginx平台初探(100%) — Nginx开发从入门到精通 http://ten 众所周知,nginx性能高,而nginx的高性能与其架构是分不开的.那么nginx究竟是怎么样的呢?这一节我们先来 ...
- Tornado异步非阻塞的使用以及原理
Tornado 和现在的主流 Web 服务器框架(包括大多数 Python 的框架)有着明显的区别:它是非阻塞式服务器,而且速度相当快.得利于其 非阻塞的方式和对 epoll 的运用,Tornado ...
- Tornado的异步非阻塞
阻塞和非阻塞Web框架 只有Tornado和Node.js是异步非阻塞的,其他所有的web框架都是阻塞式的. Tornado阻塞和非阻塞两种模式都支持. 阻塞式: 代表:Django.Flask.To ...
- Flask实现异步非阻塞请求功能
pip install gevent 关于gevent Gevent 是一个 Python 并发网络库,它使用了基于 libevent 事件循环的 greenlet 来提供一个高级同步 API.下面是 ...
- 03: 自定义异步非阻塞tornado框架
目录:Tornado其他篇 01: tornado基础篇 02: tornado进阶篇 03: 自定义异步非阻塞tornado框架 04: 打开tornado源码剖析处理过程 目录: 1.1 源码 1 ...
- Tornado 异步非阻塞
1 装饰器 + Future 从而实现Tornado的异步非阻塞 class AsyncHandler(tornado.web.RequestHandler): @gen.coroutine def ...
- Tornado之自定义异步非阻塞的服务器和客户端
一.自定义的异步非阻塞的客户端 #!/usr/bin/env python # -*- coding: utf8 -*- # __Author: "Skiler Hao" # da ...
随机推荐
- android网络监听
http://blog.csdn.net/mxiaoyem/article/details/50857008 http://blog.csdn.net/ke1vin/article/details/5 ...
- Leetcode- Find Minimum in Rotated Sorted Array-ZZ
http://changhaz.wordpress.com/2014/10/15/leetcode-find-minimum-in-rotated-sorted-array/ Suppose a so ...
- FZEasyFile的使用
FZEasyFile的使用 https://github.com/jiecao-fm/FZEasyFile 操作沙盒文件很恶心,但用上FZEasyFile就变得简单了. 以前你需要这么做才行: NSF ...
- 纯CSS画的基本图形
图形包括基本的矩形.圆形.椭圆.三角形.多边形,也包括稍微复杂一点的爱心.钻石.阴阳八卦等.当然有一些需要用到CSS3的属性,所以在你打开这篇文章的时候,我希望你用的是firefox或者chrome, ...
- dubbox源码分析(一)-服务的启动与初始化
程序猿成长之路少不了要学习和分析源码的.最近难得能静得下心来,就针对dubbox为目标开始进行源码分析. [服务提供方] 步骤 调用顺序 备注 容器启动 com.alibaba.dubbo.conta ...
- xise官方网站|xise最新版下载|-xise
诠释: 1. 破解VIP登陆限制 2.去后门 (自查) 下载地址 :https://pan.baidu.com/s/1eR2rUOM 查毒地址:http://a.virscan.org/a3983f3 ...
- TaskScheduler内幕天机解密:Spark shell案例运行日志详解、TaskScheduler和SchedulerBackend、FIFO与FAIR、Task运行时本地性算法详解等
本课主题 通过 Spark-shell 窥探程序运行时的状况 TaskScheduler 与 SchedulerBackend 之间的关系 FIFO 与 FAIR 两种调度模式彻底解密 Task 数据 ...
- 发布Hessian服务作为服务内部基础服务
摘要:Hessian经常作为服务内部RPC工具来使用,速度快效率高.重构代码的核心思想就是把共用的代码段提出来,使代码结构优化:架构设计类似,把基本的共用的服务提出来,使架构优化.下面讲述一下我在具体 ...
- MQ7.5以后的权限问题解决
MQ7.5以后权限是个问题,目前我也没有什么特别好的解决办法,把认证通道关闭就可以正常使用. 下面是IBM 官方的解释,可惜我没调通,望高人指点! 疑问 您使用MQ 7.1或者7.5创建了一个新的队列 ...
- thinkPHP输出sql语句(3.2和5.0通用)
//5.0$qwe = db::table('think_user')->where('id',1)->fetchsql()->column('name'); dump($qwe); ...