线程和进程的操做是由程序触发系统接口,最后的执行者是系统;协程的操做则是程序员。python
协程存在的意义:对于多线程应用,CPU经过切片的方式来切换线程间的执行,线程切换时须要耗时(保存状态,下次继续)。协程,则只使用一个线程,在一个线程中规定某个代码块执行顺序。git
协程的适用场景:当程序中存在大量不须要CPU的操做时(IO),适用于协程;程序员
event loop是协程执行的控制点, 若是你但愿执行协程, 就须要用到它们。github
event loop提供了以下的特性:多线程
一、注册、执行、取消延时调用(异步函数)
二、建立用于通讯的client和server协议(工具)
三、建立和别的程序通讯的子进程和协议(工具)
四、把函数调用送入线程池中异步
协程示例一: async
1 #!/usr/bin/env python 2 # -*- coding:utf-8 -*- 3 """ 4 协程示例 5 """ 6 import asyncio 7 8 async def test1(): 9 print("test1, starting") 10 await test2() 11 print("test1, ending") 12 13 async def test2(): 14 print("test2 start") 15 16 loop = asyncio.get_event_loop() #asyncio.get_event_loop() : asyncio启动默认的event loop 17 loop.run_until_complete(test1()) #run_until_complete() : 这个函数是阻塞执行的,知道全部的异步函数执行完成, 18 loop.close() #close() : 关闭event loop。
greenlet协程示例:函数
1 #!/usr/bin/env python 2 # -*- coding:utf-8 -*- 3 """ 4 协程示例 5 """ 6 import greenlet 7 8 def fun1(): 9 print("12") 10 f2.switch() 11 print("56") 12 f2.switch() 13 14 def fun2(): 15 print("34") 16 f1.switch() 17 print("78") 18 19 20 f1 = greenlet.greenlet(fun1) 21 f2 = greenlet.greenlet(fun2) 22 23 f1.switch()
gevent协程示例:工具
1 #!/usr/bin/env python 2 # -*- coding:utf-8 -*- 3 """ 4 gevent协程示例 5 """ 6 import gevent 7 8 def fun1(): 9 print("www.baidu.com") #第一步 10 gevent.sleep(0) 11 print("end the baidu.com") #第三步 12 13 def fun2(): 14 print("www.yusheng.com") #第二步 15 gevent.sleep(0) 16 print("end the yusheng.com") #第四步 17 18 gevent.joinall([ 19 gevent.spawn(fun1), 20 gevent.spawn(fun2), 21 ])
示例三:遇到IO操做自动切换:oop
1 #!/usr/bin/env python 2 # -*- coding:utf-8 -*- 3 """ 4 遇到IO操做自动切换: 5 """ 6 import gevent 7 import requests 8 9 def fun(url): 10 print("get: %s" % url) 11 gevent.sleep(0) 12 data = requests.get(url) 13 ret = data.text 14 print(url, len(ret)) 15 16 gevent.joinall([ 17 gevent.spawn(fun, 'https://www.python.org/'), 18 gevent.spawn(fun, 'https://www.yahoo.com/'), 19 gevent.spawn(fun, 'https://github.com/'), 20 ])