资讯专栏INFORMATION COLUMN

Python:Tornado 第一章:异步及协程基础:第三节:协程

charles_paul / 436人阅读

摘要:上一篇文章第一章异步及协程基础第二节关键字下一篇文章第二章实战演练开发网站第一节网站结构使用协程可以开发出类似同步代码的异步行为。协程函数可以通过以下三张方式调用在本身是协程的函数内通过关键字调用。

上一篇文章:Python:Tornado 第一章:异步及协程基础:第二节:Python关键字yield
下一篇文章:Python:Tornado 第二章:实战演练:开发Tornado网站:第一节:网站结构:HelloWorld

使用Tornado协程可以开发出类似同步代码的异步行为。同时,因为协程本身不使用线程,所以减少了线程上下文切换的开销,是一种高效的开发模式。

1、编写协程函数

实例:用协程技术开发网页访问功能

#用协程技术开发网页访问功能
from tornado import  gen #引入协程库gen
from tornado.httpclient import AsyncHTTPClient
import time

#使用gen.coroutine修饰器
@gen.coroutine
def coroutine_visit():
    http_client=AsyncHTTPClient()
    response=yield http_client.fetch("http://www.baidu.com")
    print(response.body)

本例中任然使用了异步客户端AsyncHTTPClient进行页面访问,装饰器@gen.coroutine声明这是一个协程函数,由于yield关键字的作用,使得代码中不用再编写回调函数用于处理访问结果,而可以直接在yield语句的后面编写结果处理语句。

2、调用协程函数

由于Tornado协程基于Python的yield关键字实现,所以不能像普通函数那样直接调用。
协程函数可以通过以下三张方式调用:

在本身是协程的函数内通过yield关键字调用。

在IOLoop尚未启动时,通过IOLoop的run_sync()函数调用。

在IOLoop已经启动时,通过IOLoop的spawn_callback()函数调用。

实例:通过协程函数调用协程函数

代码:

#用协程技术开发网页访问功能
from tornado import  gen #引入协程库gen
from tornado.httpclient import AsyncHTTPClient
import time

#使用gen.coroutine修饰器
@gen.coroutine
def coroutine_visit():
    http_client=AsyncHTTPClient()
    response=yield http_client.fetch("http://www.baidu.com")
    print(response.body)

@gen.coroutine
def outer_coroutine():
    print("start call coroutine_visit")
    yield coroutine_visit()
    print("end call coroutine_cisit")

本例中outer_coroutine()和coroutine_visit()都是协程函数,所以他们之间可以通过yield关键字调用。_

实例:IOLoo尚未启动时,通过IOLoop的run_sync()函数调用。
IOLoop是Tornado的主事件循环对象,Tornado程序通过它监听外部客户端的访问请求,并执行相应操作。

代码:

#用协程技术开发网页访问功能
from tornado import  gen #引入协程库gen
from tornado.httpclient import AsyncHTTPClient
from tornado.ioloop import IOLoop  #引入IOLoop对象

#使用gen.coroutine修饰器
@gen.coroutine
def coroutine_visit():
    http_client=AsyncHTTPClient()
    response=yield http_client.fetch("http://www.baidu.com")
    print(response.body)

def func_normal():
    print("start call coroutine_visit")
    IOLoop.current().run_sync(lambda :coroutine_visit())
    print("end call coroutine_visit")
当程序尚未进入IOLoop的running状态时,可以通过run_sync()函数调用协程函数。

⚠️注意:run_sync()函数将阻塞当前函数的调用,直到被调用的协程执行完成。

事实上,Tornado要求协程函数在IOLoop的running状态种才能被调用,只不过run_sync函数自动完成了启动、停止IOLoop的操作步骤,他的实现逻辑是:

【启动IOLoop】》【调用被lambda封装的协程函数】》【停止IOLoop】

实例:在IOLoop启动时,通过spawn_callback()函数调用

代码:

#用协程技术开发网页访问功能
from tornado import  gen #引入协程库gen
from tornado.httpclient import AsyncHTTPClient
from tornado.ioloop import IOLoop  #引入IOLoop对象

#使用gen.coroutine修饰器
@gen.coroutine
def coroutine_visit():
    http_client=AsyncHTTPClient()
    response=yield http_client.fetch("http://www.baidu.com")
    print(response.body)

def func_normal():
    print("start call coroutine_visit")
    IOLoop.current().spawn_callback(coroutine_visit)
    print("end call coroutine_visit")
spawn_callback()函数将不会等待被调用协程执行完成,所有上下两条打印语句将马上完成,而coroutine__visit本身将会由IOLoop在合适的时机进行调用。

⚠️注意:IOLoop的spawn_callback()函数没有为开发者提供获取协程函数调用返回值的方法,所以只能用span_callback()调用没有返回值的协程函数。

3、在协程中调用阻塞函数
在协程中直接调用阻塞函数会影响协程本身的性能,所以Tornado提供了在协程中利用线程池调度阻塞函数,从而不影响协程本身继续执行的方法。

代码实例:

from concurrent.futures import ThreadPoolExecutor
from tornado import gen

#定义线程池
thread_pool=ThreadPoolExecutor(2)

def mySleep(count):
    import time
    for x in range(count):
        time.sleep(1)

@gen.coroutine
def call_blocking():
    print("start")
    yield thread_pool.submit(mySleep,10)
    print("end")

代码中首先引用了concurrent.futures种的ThreadPoolExecutor类,实例化了一个由两个线程的线程池thread_pool。在需要调用阻塞函数的协程call_blocking种,使用thread_pool.submit调用阻塞函数,并通过yield返回。这样便不会阻塞协程所在的线程的继续执行,也保证了阻塞函数前后代码的执行顺序。

4、在协程中等待多个异步调用
到目前为止,我们知道了协程中一个yield关键字等待一个异步调用的编程方法。其实,Tornado允许在协程中用一个yield关键字等待多个异步调用,只需要把这些调用以列表(list)或字典(dictionary)的方式传递给yield关键字即可。
实例:使用列表方式传递多个异步调用
#使用列表方式传递多个异步调用
from tornado import gen  #引入协程库gen
from tornado.httpclient import AsyncHTTPClient

@gen.coroutine   #使用gen.coroutine修饰器
def coroutine_visit():
    http_client=AsyncHTTPClient()
    list_response=yield [
        http_client.fetch("http://www.baidu.com"),
        http_client.fetch("http://www.api.jiutouxiang.com")
    ]
    for response in list_response:
        print(response.body)


在代码中仍然使用@gen.coroutine装饰器定义协程,在需要yield的地方用列表传递若干个异步调用,只有在列表种的所有调用都执行完成后,yield才会返回并且继续执行。yield以列表方式返回调用结果。

实例:用字典方式传递多个异步调用:
#使用列表方式传递多个异步调用
from tornado import gen  #引入协程库gen
from tornado.httpclient import AsyncHTTPClient

@gen.coroutine   #使用gen.coroutine修饰器
def coroutine_visit():
    http_client=AsyncHTTPClient()
    dict_response=yield {
       "baidu": http_client.fetch("http://www.baidu.com"),
        "9siliao":http_client.fetch("http://www.api.jiutouxiang.com")
    }
    print(dict_response["baidu"].body)

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/42598.html

相关文章

  • PythonTornado 一章异步协程基础:第二节:Python关键字yield

    摘要:在种,使用关键字定义的迭代器也被称为生成器迭代器迭代器是访问集合内元素的一种方式。调用任何定义包含关键字的函数都不会执行该函数,而是会获得一个队应于该函数的迭代器。 上一篇文章:Python:Tornado 第一章:异步及协程基础:第一节:同步与异步I/O下一篇文章:Python:Tornado 第一章:异步及协程基础:第三节:协程 协程是Tornado中进行异步I/O代码开发的方法...

    reclay 评论0 收藏0
  • PythonTornado 一章异步协程基础第一节:同步与异步I/O

    摘要:上一篇文章开篇下一篇文章第一章异步及协程基础第二节关键字协程是种推荐的编程方式,使用协程可以开发出简捷高效的异步处理代码。同步操作,导致进程阻塞,直到操作完成异步操作,不会导致请求进程阻塞。 上一篇文章:Python:Tornado 开篇下一篇文章:Python:Tornado 第一章:异步及协程基础:第二节:Python关键字yield 协程是Tornado种推荐的编程方式,使用协...

    Anleb 评论0 收藏0
  • PythonTornado 第二章:实战演练:开发Tornado网站:第一节:网站结构:Hello

    摘要:上一篇文章第一章异步及协程基础第三节协程下一篇文章第二章实战演练开发网站第二节网站结构路由解析实例浏览器输入链接页面显示下面逐行解析上面的代码做了些什么首先通过语句引入包中的和类。该对象的第一个餐食用于定义程序的路由映射。 上一篇文章:Python:Tornado 第一章:异步及协程基础:第三节:协程下一篇文章:Python:Tornado 第二章:实战演练:开发Tornado网站:第...

    Taonce 评论0 收藏0
  • PythonTornado 开篇

    摘要:作为网站的基础框架,于年月日发布,目前已经获得了很多社区的支持,并且在一系列不同的场景种得到应用。使用该框架,开发者能够快速开发出即安全又强大的用户身份认证机制,例如机制用户身份认证防止跨站攻击等等。 下一篇文章:Python:Tornado 第一章:异步及协程基础:第一节:同步与异步I/O Tornado是一个可扩展的非阻塞Web服务器以及相关工具的总称。Tornado每秒可以处理...

    Anshiii 评论0 收藏0
  • Tornado 4.3文档翻译: 用户指南-协程

    摘要:译者说于年月日发布,该版本正式支持的关键字,并且用旧版本编译同样可以使用这两个关键字,这无疑是一种进步。其次,这是最后一个支持和的版本了,在后续的版本了会移除对它们的兼容。 译者说 Tornado 4.3于2015年11月6日发布,该版本正式支持Python3.5的async/await关键字,并且用旧版本CPython编译Tornado同样可以使用这两个关键字,这无疑是一种进步。其次...

    SimonMa 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<