一:使用 asyncio处理并发

介绍 asyncio 包,这个包使用事件循环驱动的协程实现并发。这是 Python 中最大也是最具雄心壮志的库之一。

二:示例

  1)单任务协程处理和普通任务比较

#普通任务示例
# _*_ coding:utf-8 _*_
__author__ = "lixiang"
import asyncio
import time def hello():
count=0
while count<10:
print("before",count)
try: time.sleep(0.1)
count+=1
print("after",count)
except Exception as e:
print(e)
return count if __name__=="__main__":
start_time=time.time()
hello()
end_time=time.time()
print("cost time",end_time-start_time)

普通任务示例

#单任务协程示例
# _*_ coding:utf-8 _*_
__author__ = "lixiang"
import asyncio
import time
@asyncio.coroutine# 表示要协程处理
def hello():
count=0
while count<10:
print("before",count)
try:
#使用 yield from asyncio.sleep(1) 代替 time.sleep(1),
yield from asyncio.sleep(0.1)
count+=1
print("after",count)
except Exception as e:
print(e)
return count if __name__=="__main__":
start_time=time.time()
loop=asyncio.get_event_loop()
loop.run_until_complete(hello())
loop.close()
end_time=time.time()
print("cost time",end_time-start_time)

单任务协程示例

  两种方式处理结果处理时间都是1秒,此时协程并不能体现并发的优势

# _*_ coding:utf-8 _*_
__author__ = "lixiang"
import asyncio
import time
@asyncio.coroutine# 要协程处理
def hello():
count=0
while count<10:
print("before",count)
try:
#使用 yield from asyncio.sleep(0.1) 代替 time.sleep(0.1),这样的休眠不会阻塞事件循环
yield from asyncio.sleep(0.1)
count+=1
print("after",count)
except Exception as e:
print(e)
return count @asyncio.coroutine
def hello_welcome():
result=yield from hello()
print(result) if __name__=="__main__":
start_time=time.time()
loop=asyncio.get_event_loop()
loop.run_until_complete(hello_welcome())
loop.close()
end_time=time.time()
print("cost time",end_time-start_time)

通过一个协程直接调用另外一协程

  此时也没有显示协程优势

  2)两个任务协程比较

  上面示例,如果hello运行需要1秒,别外一个任务运行也需要1秒。

# _*_ coding:utf-8 _*_
__author__ = "lixiang"
import asyncio
import time
@asyncio.coroutine# 要协程处理
def hello():
count=0
while count<10:
print("before",count)
try:
#使用 yield from asyncio.sleep(0.1) 代替 time.sleep(0.1),这样的休眠不会阻塞事件循环
yield from asyncio.sleep(0.1)
count+=1
print("after",count)
except Exception as e:
print(e)
return count @asyncio.coroutine
def show_hello():
yield from asyncio.sleep(1)
return 666 @asyncio.coroutine
def hello_welcome():
"""
安排协程hello的运行时间,这个运行时候根据show_hello来定。
如果show_hello安排时间为0.5秒,hello时间为0.1秒,10次循环不能正常结束,
提示Task was destroyed but it is pending!
如果show_hello安排时间为1秒,hello时间为0.1秒,10次循环能正常结束,
任务提示:Task finished
"""
re1=asyncio.async(hello())
result=yield from show_hello()
print(re1)
print("result",result) if __name__=="__main__":
start_time=time.time()
loop=asyncio.get_event_loop()
loop.run_until_complete(hello_welcome())
loop.close()
end_time=time.time()
print("cost time",end_time-start_time) #测试结果显示只需要1秒,并发优势显示出来。
这里我们对hello进行协程运行时间安排。
安排协程hello的运行时间,这个运行时间根据show_hello来定。
如果show_hello安排时间为0.5秒,hello时间为0.1秒,10次循环不能正常结束,
提示Task was destroyed but it is pending!
如果show_hello安排时间为0.9秒,hello时间为0.1秒,10次循环能正常结束,
任务提示:Task finished

两个任务协程比较

#多任务
# _*_ coding:utf-8 _*_
__author__ = "lixiang"
import asyncio
import time
@asyncio.coroutine# 要协程处理
def hello():
count=0
while count<10:
print("before",count)
try:
#使用 yield from asyncio.sleep(.1) 代替 time.sleep(.1),这样的休眠不会阻塞事件循环
yield from asyncio.sleep(0.1)
count+=1
print("after",count)
except Exception as e:
print(e)
return count @asyncio.coroutine
def show_hello():
yield from asyncio.sleep(1)
return 666 @asyncio.coroutine
def show_hello2():
yield from asyncio.sleep(1)
return 777 @asyncio.coroutine
def hello_welcome():
"""
安排协程hello的运行时间,这个运行时候根据show_hello来定。
如果show_hello安排时间为0.5秒,hello时间为0.1秒,10次循环不能正常结束,
提示Task was destroyed but it is pending!
如果show_hello安排时间为0.9秒,hello时间为0.1秒,10次循环能正常结束,
任务提示:Task finished
"""
asyncio.async(show_hello2())
asyncio.async(hello())
result=yield from show_hello() print("result",result) if __name__=="__main__":
start_time=time.time()
loop=asyncio.get_event_loop()
loop.run_until_complete(hello_welcome())
loop.close()
end_time=time.time()
print("cost time",end_time-start_time)

多任务

    @asyncio.coroutine# 要协程处理
def hello():
count=0
while count<10:
print("before",count)
try:
#使用 yield from asyncio.sleep(.1) 代替 time.sleep(.1),这样的休眠不会阻塞事件循环
yield from asyncio.sleep(0.1)
count+=1
print("after",count)
yield from asyncio.sleep(0.1)
except Exception as e:
print(e)

如果 hello里面有两个yield,两者之间是顺序,机每次循环需要0.2

上面为了执行这些操作,必须排定协程的运行时间,然后使用 asyncio.Task 对象包装协
    程。对协程来说,获取 Task 对象有两种主要方式。

   asyncio.async(coro_or_future, *, loop=None)

  create_

  上面如果都是通过时间控制协程运行,那如果控制协程的时间怎么设定比较好了,

  时间短了其他协程任务不能运行结束Task was destroyed but it is pending,时间设置长又会耗时长了。
     可以通过asyncio.wait解决
       asyncio.wait(...) 协程的参数是一个由期物或协程构成的可迭代对象; wait 会分别
       把各个协程包装进一个 Task 对象。最终的结果是, wait 处理的所有对象都通过某种方
      式变成 Future 类的实例。 wait 是协程函数,因此返回的是一个协程或生成器对
      象; wait_coro 变量中存储的正是这种对象。为了驱动协程,我们把协程传给
      loop.run_until_complete(...) 方法。虽然函数的名称是 wait,但它不是阻塞型函数。

   wait 是一个协程,等传给它的所有协程运行完毕后结束(这是 wait 函数的默认行为;参见这个示例后面的说明)
    loop.run_until_complete 方法的参数是一个期物或协程。如果是协
    程, run_until_complete 方法与 wait 函数一样,把协程包装进一个 Task 对象中。协
    程、期物和任务都能由 yield from 驱动,这正是 run_until_complete 方法对 wait
    函数返回的 wait_coro 对象所做的事。 wait_coro 运行结束后返回一个元组,第一个元
    素是一系列结束的期物,第二个元素是一系列未结束的期物

    # _*_ coding:utf-8 _*_
__author__ = "lixiang"
import asyncio
import time
@asyncio.coroutine# 要协程处理
def hello():
count=0
while count<10:
print("before",count)
try:
#使用 yield from asyncio.sleep(.1) 代替 time.sleep(.1),这样的休眠不会阻塞事件循环
yield from asyncio.sleep(0.1)
count+=1
print("after",count)
yield from asyncio.sleep(0.1)
except Exception as e:
print(e)
return count @asyncio.coroutine
def show_hello():
yield from asyncio.sleep(1.2)
return 666 @asyncio.coroutine
def show_hello2():
yield from asyncio.sleep(1.2)
return 777 # @asyncio.coroutine
# def hello_welcome():
# """
# 安排协程hello的运行时间,这个运行时候根据show_hello来定。
# 如果show_hello安排时间为0.5秒,hello时间为0.1秒,10次循环不能正常结束,
# 提示Task was destroyed but it is pending!
# 如果show_hello安排时间为0.9秒,hello时间为0.1秒,10次循环能正常结束,
# 任务提示:Task finished
# """
# asyncio.async(show_hello2())
# asyncio.async(hello())
# result=yield from show_hello()
#
# print("result",result) if __name__=="__main__":
start_time=time.time()
loop=asyncio.get_event_loop()
wait_coro=asyncio.wait([hello(),show_hello(),show_hello2()])
loop.run_until_complete(wait_coro)
wait_coro.close()
loop.close()
end_time=time.time()
print("cost time",end_time-start_time)

asyncio wait使用

#如果要实时显示那个协程已完成了
    我把一个协程列表传给 asyncio.wait 函数,经由
    loop.run_until_complete 方法驱动,全部协程运行完毕后,这个函数会返回所有下载
    结果。可是,为了更新进度条,各个协程运行结束后就要立即获取结果。在线程池版示例
    中(见示例 17-14),为了集成进度条,我们使用的是 as_completed 生成器函数;幸
    好, asyncio 包提供了这个生成器函数的相应版本

# _*_ coding:utf-8 _*_
__author__ = "lixiang"
import asyncio
import time
@asyncio.coroutine# 要协程处理
def hello():
count=0
while count<10:
print("before",count)
try:
#使用 yield from asyncio.sleep(.1) 代替 time.sleep(.1),这样的休眠不会阻塞事件循环
yield from asyncio.sleep(0.1)
count+=1
print("after",count)
yield from asyncio.sleep(0.1)
except Exception as e:
print(e)
return count @asyncio.coroutine
def show_hello():
yield from asyncio.sleep(1.2)
return 666 @asyncio.coroutine
def show_hello2():
yield from asyncio.sleep(1.3)
return 777 @asyncio.coroutine
def hello_welcome():
count=0
coro=asyncio.as_completed([hello(),show_hello(),show_hello2()]) for future in coro:
res=yield from future
print("",res)
count+=1
print(count)#可以在这里实现进度条功能
return count if __name__=="__main__":
start_time=time.time()
loop=asyncio.get_event_loop()
counts=loop.run_until_complete(hello_welcome())
print(counts)
loop.close()
end_time=time.time()
print("cost time",end_time-start_time)

asyncio as_complated

6)协程三( asyncio处理并发)的更多相关文章

  1. python并发编程之asyncio协程(三)

    协程实现了在单线程下的并发,每个协程共享线程的几乎所有的资源,除了协程自己私有的上下文栈:协程的切换属于程序级别的切换,对于操作系统来说是无感知的,因此切换速度更快.开销更小.效率更高,在有多IO操作 ...

  2. Python进阶----异步同步,阻塞非阻塞,线程池(进程池)的异步+回调机制实行并发, 线程队列(Queue, LifoQueue,PriorityQueue), 事件Event,线程的三个状态(就绪,挂起,运行) ,***协程概念,yield模拟并发(有缺陷),Greenlet模块(手动切换),Gevent(协程并发)

    Python进阶----异步同步,阻塞非阻塞,线程池(进程池)的异步+回调机制实行并发, 线程队列(Queue, LifoQueue,PriorityQueue), 事件Event,线程的三个状态(就 ...

  3. 【python】-- 协程介绍及基本示例、协程遇到IO操作自动切换、协程(gevent)并发爬网页

    协程介绍及基本示例 协程,又称微线程,纤程.英文名Coroutine.一句话说明什么是协程:协程是一种用户态的轻量级线程. 协程拥有自己的寄存器上下文和栈.协程调度切换时,将寄存器上下文和栈保存到其他 ...

  4. 一个有趣的小例子,带你入门协程模块-asyncio

    一个有趣的小例子,带你入门协程模块-asyncio 上篇文章写了关于yield from的用法,简单的了解异步模式,[https://www.cnblogs.com/c-x-a/p/10106031. ...

  5. Python协程与asyncio

    asyncio(解决异步io编程的一整套解决方案,它主要用于异步网络操作.并发和协程)协程(Coroutine一种用户态的轻量级微线程,它是程序级别的,在执行过程中可以中断去执行其它的子程序,别的子程 ...

  6. Kotlin Coroutine(协程): 三、了解协程

    @ 目录 前言 一.协程上下文 1.调度器 2.给协程起名 3.局部变量 二.启动模式 CoroutineStart 三.异常处理 1.异常测试 2.CoroutineExceptionHandler ...

  7. 协程+IO切换实现并发

    from gevent import monkey # 以后代码中遇到IO都会自动执行greenlet的switch进行切换 monkey.patch_all() import requests im ...

  8. 利用协程和socket实现并发

    服务端代码 from gevent import monkey monkey.patch_all() from gevent import spawn import socket def commun ...

  9. python协程--asyncio模块(基础并发测试)

    在高并发的场景下,python提供了一个多线程的模块threading,但似乎这个模块并不近人如意,原因在于cpython本身的全局解析锁(GIL)问题,在一段时间片内实际上的执行是单线程的.同时还存 ...

随机推荐

  1. DevExpress 之 GridControl 自定义列

    Ø  前言 DevExpress 控件大家应该都有所了解,使用这个框架实现B/S或C/S的,都是非常出色的.本文主要讨论下 GridControl 中如何[自定义列]或[计算列],可使用以下两种方法实 ...

  2. js apply使用

    js中apply方法的使用 1.对象的继承,一般的做法是复制:Object.extend prototype.js的实现方式是:  Object.extend = function(destinati ...

  3. kindeditor<=4.1.5 文件上传漏洞利用

    kindeditor<=4.1.5 文件上传漏洞 - Kindeditor <=4.1.5 file upload vulnerability and use 漏洞存影响版本:小于等于4. ...

  4. [C++]线性链表之单链表

    [文档整理系列] 线性链表之单链表 /* 问题描述:线性表____链表_____单链表 @date 2017-3-7 */ #include<iostream> using namespa ...

  5. 第28月第22天 iOS动态库

    1. NIMSDK 在 5.1.0 版本之后已改为动态库,集成方式有所改变,若需要集成高于此版本的 SDK,只需要做以下步骤: 将下载的 SDK 拖动到 Targets -> General - ...

  6. Django REST framework 第四章 Authentication

    到目前为止,撰写的API没有任何限制关于谁能更新.删除snippet. 我们更想要一些高级行为来确保: 1.代码段总是跟创建者有关联 2.只要认证通过的用户才能创建 3.只有创建者有权限更新或者删除 ...

  7. 解释下面URL

    解释下面URL各部分的含义 a.duke.csc.villanova.edu/jss/examles.html duke是计算机名,该计算机属于villanova.edu域的csc子域.edu是最高级 ...

  8. linux下安装oh-my-zsh

    如果是linux 系统,首先你需要安装 zsh sudo yum install zsh 或者 sudo apt-get install zsh 接下来我们需要下载 oh-my-zsh 项目来帮我们配 ...

  9. 用NAME_N带入NAME 让显示格式变为 姓名(类型),类型在数据库中是1和0,显示效果为姓名(1),SQL写法

    select xxxx T.PROJECT_NAME||'('||DECODE(T.PROJECT_TYPE,'1','收入','2','支出','3','挂账')||')' PROJECT_NAME ...

  10. iframe教程

    有关iframe的最强大的强大的教程 $(window.parent.document).contents().find("#tab_release"+taskId2+" ...