Python协程、线程
看了很多文章说线程、协程,讲的真是五花八门,新人听不懂类型,就算听懂了,也写不会类型。看得我脑壳疼。相信线程的话,是非常清晰明了的,究竟什么是协程呢?
还是那句话,不管他是个什么玩意儿, 首先你知道怎么用,知道怎么用了之后,再去深究他是个什么玩意儿的时候就非常的清晰明了。
看懂线程、协程必须具备的思想
不要把线程、协程想的有多么困难,多么复杂,其实就是异步的使用工具而已。看待任何问题,一定要站在比问题本身高一个层次再去理解问题。
什么是协程
协程是运行在线程之中的,就像线程在进程中一样。协程的使用表现在一个线程中规定代码块的执行顺序。
不同点
1、上下文的保存机制不同:
线程:上下文保存在任务控制块(TCB)中,故每次切换都会消耗一定量的资源。
协程:自己的上下文管理区域,故切换消耗的资源是非常小的,几乎可以忽略不计。
资源消耗问题是使用协程的主要原因所在。你只需要知道基本原理即可,切不必深究太多,太底层会把你整懵逼。
2、调度不同:
线程:系统自动调度,分为两种方式:
1、抢占式调度:操作系统会轮流询问每一个任务是否需要使用 CPU ,需要使用的话就让它用,不过在一定时间后,操作系统会剥夺当前任务的 CPU 使用权,再去询问下一个任务。如果有一个任务死锁,系统仍能正常运行。
2、协同式调度:就像接力赛,一个跑完接下一个。
协程:协作式任务,系统完全霸占cpu,除非自己放弃使用(协程耗时等待的情况下才会切换)。这也是为什么协程中只要一个地方死锁,整个程序就死锁的原因。
代码
import asyncio async def work(x): await asyncio.sleep(x) #IO阻塞 return x async def main(): #创建协程任务 coroutine1 = work(1) coroutine2 = work(2) coroutine3 = work(4) tasks = [ asyncio.ensure_future(coroutine1), asyncio.ensure_future(coroutine2), asyncio.ensure_future(coroutine3) ] return await asyncio.wait(tasks) loop = asyncio.get_event_loop() #创建事件循环,用于协程间来回切换 done, pending = loop.run_until_complete(main()) #执行 #done是返回值 for task in done: print('Task Result: ', task.result())