Python的协程

什么是协程?

协程,本质上还是单进程单线程运行的程序,并不能提升运算速度,但是适用于网络通讯,因为网络通讯中,经常会碰到等待的情况。

asyncio

看下面一段代码:

import asyncio
async def main():
	print('hello')
	await asyncio.sleep(1)
	print('world')
coro = main()
asyncio.run(main())
  • event_loop:核心组件,相当于一个大脑,他来决定该执行哪个任务。就像上面说到的,同时执行的任务只能有一个,不存在系统级的上下文切换。
  • task:event_loop来执行
  • coroutine:分为coroutine function、coroutine object:
    上面的main函数就是一个coroutine function,由async修饰。不像其他函数,这个被async修饰的函数,被调用的时候返回的是一个coroutine object,也就是coro变量是一个coroutine function,执行coro赋值,并不会打印“hello world”。 那我们要怎么运行coroutine object呢?上面的代码中,asyncio.run(main())就是把coroutine object变成task交由event_loop来执行,得到结果。

再来看这段代码:

import asyncio
import time

async def say_after(delay, what):
	await asyncio.sleep(delay)
	print(what)

async def main():
	print(f"started at {time.strftime('%X')}")
	await say_after(1, 'hello')
	await say_after(2, 'hello')
	print(f"finished at {time.strftime('%X')}")
asyncio.run(main())

这一段会执行3秒,为什么呢?await也会把一个coroutine object变成task,交由event_loop来执行,并返回结果。event_loop不能从task中抢占使用权,只有等task执行结束或者调用await的时候才会返回控制权。上面这段代码和不使用await,async关键词的效果是一样的,他的运行过程如下:首先asyncio.run()会把main函数变成一个task交由event_loop来执行,执行过程中,遇到say_after(1, ‘hello’),await又把say_after(1, ‘hello’)变成一个task放到event_loop中,这个时候就执行say_after(1, ‘hello’),遇到asyncio.sleep(delay),那么await再把asyncio.sleep(delay)变成task,放到event_loop中。要继续执行say_after(1, ‘hello’),要等asyncio.sleep(delay)完成,那么我们就执行asyncio.sleep(delay),执行完,在执行say_after(1, ‘hello’),打印输出。就相当于一个函数栈。

还有一段代码:

import asyncio
import time

async def say_after(delay, what):
	await asyncio.sleep(delay)
	print(what)

async def main():
	task1 = asyncio.create_task(say_after(1, 'hello'))
	task2 = asyncio.create_task(say_after(2, 'hello'))
	print(f"started at {time.strftime('%X')}")
	await task1
	await task2
	print(f"finished at {time.strftime('%X')}")
	
asyncio.run(main())

执行到task1赋值时,控制权依然在event_loop中,执行完task1赋值,还能继续执行task2创建,创建完task2,这个时候event_loop中有两个task了。直到运行到await task1时,提交任务,执行函数。在函数体内部await asyncio.sleep(delay)时,需要等待,这个时候event_loop发现任务队列中还有一个任务等着执行,那么他就去执行另一个任务,也就是task2。因此,这一段代码运行只需要2秒。

这两段代码的区别就是,在执行task2,也就是say_after(2, ‘hello’),event_loop中是否已经知道了有这个任务。如果没有,那么就不会再task1等待的时候调过来执行say_after2,如果有,就会。

总结

  1. 为什么要协程?

线程之间的切换,线程之间的切换也是需要时间的。而协程是基于单线程单进程的思想构建出来的,这能够解决线程切换耗时的问题(相对来说),减少系统开销,提高运行性能。

  1. 线程适用于像网络通讯这种需要等待I/O的任务。
  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
协程Coroutine)是一种用户态的轻量级线程,它可以在单个线程中实现多任务并发处理。Python中的协程通过生成器(generator)实现,使用yield语句来实现协程的暂停和恢复操作。 在Python 3.5之后,Python引入了async/await关键字,使得协程的使用更加方便和简洁。 下面是一个使用yield实现协程的示例: ```python def coroutine(): print("coroutine started") while True: value = yield print("Received value: ", value) c = coroutine() next(c) # 启动协程 c.send(1) # 发送值,并打印接收到的值 c.send(2) ``` 输出: ``` coroutine started Received value: 1 Received value: 2 ``` 在上面的代码中,使用yield语句实现了协程的暂停和恢复操作。在调用`c.send()`方法时,会将值发送给协程,并从yield语句处恢复协程的执行。协程会处理接收到的值,并在下一个yield语句处暂停,等待下一次发送。 除了使用yield语句来实现协程外,Python 3.5之后还可以使用async/await关键字来定义协程。使用async/await关键字定义的协程更加简洁和易于理解。下面是一个使用async/await关键字实现的协程示例: ```python async def coroutine(): print("coroutine started") while True: value = await asyncio.sleep(1) print("Received value: ", value) asyncio.run(coroutine()) ``` 在上面的代码中,使用async/await关键字定义了一个协程。使用`asyncio.sleep()`函数来实现协程的暂停操作,并在下一次事件循环时恢复协程的执行。使用`asyncio.run()`函数来运行协程。 总的来说,协程是一种非常有用的并发编程技术,可以在单个线程中实现高并发的处理。在Python中,可以使用生成器和async/await关键字来实现协程

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值