协程是个啥玩意

Python中有一个协程的概念,那么,协程是个啥玩意呢?

协程,又称微线程,纤程。英文名Coroutine。

协程是python个中另外一种实现多任务的方式,只不过比线程更小占用更小执行单元(理解为需要的资源)。 为啥说它是一个执行单元,因为它自带CPU上下文。这样只要在合适的时机, 我们可以把一个协程 切换到另一个协程。 只要这个过程中保存或恢复 CPU上下文那么程序还是可以运行的。

通俗的理解:在一个线程中的某个函数,可以在任何地方保存当前函数的一些临时变量等信息,然后切换到另外一个函数中执行,注意不是通过调用函数的方式做到的,并且切换的次数以及什么时候再切换到原来的函数都由开发者自己确定。

协程和线程的差异

在实现多任务时, 线程切换从系统层面远不止保存和恢复 CPU上下文这么简单。 操作系统为了程序运行的高效性每个线程都有自己缓存Cache等等数据,操作系统还会帮你做这些数据的恢复操作。 所以线程的切换非常耗性能。但是协程的切换只是单纯的操作CPU的上下文,所以一秒钟切换个上百万次系统都抗的住。

实现1(yield)

import time

def a():
    while True:
        print('---1---')
        yield None
        time.sleep(0.5)

def b():
    while True:
        print('---2---')
        yield None
        time.sleep(0.5)

if __name__ == '__main__':
    t1 = a()
    t2 = b()
    while True:
        next(t1)
        next(t2)

运行结果:

循环打印

实现2(greenlet)

from greenlet import greenlet
import time

def a():
    while True:
        print('---1---')
        gr2.switch()
        time.sleep(0.5)

def b():
    while True:
        print('---2---')
        gr1.switch()
        time.sleep(0.5)

if __name__ == '__main__':
    gr1 = greenlet(a)
    gr2 = greenlet(b)

    # 切换到gr1执行
    gr1.switch()

运行结果同上。

实现3(gevent)

greenlet已经实现了协程,但是这个还的人工切换,是不是觉得太麻烦了,不要捉急,python还有一个比greenlet更强大的并且能够自动切换任务的模块,就是gevent。

其原理是当一个greenlet遇到IO(指的是input output 输入输出,比如网络、文件操作等)操作时,比如访问网络,就自动切换到其他的greenlet,等到IO操作完成,再在适当的时候切换回来继续执行。

由于IO操作非常耗时,经常使程序处于等待状态,有了gevent为我们自动切换协程,就保证总有greenlet在运行,而不是等待IO。

import gevent
from gevent import monkey
import time

#打补丁
#如果不打补丁的话,下面就必须写上gevent.sleep(0.5)
monkey.patch_all()

def a():
    while True:
        print('---1---')
        time.sleep(0.5)

def b():
    while True:
        print('---2---')
        time.sleep(0.5)

if __name__ == '__main__':
    g1 = gevent.spawn(a)
    g2 = gevent.spawn(b)

    g1.join()
    g2.join()

运行结果同上。

gevent强大在当有耗时操作的时候,会自动切换其他不耗时的协程。

import gevent
from gevent import monkey
import time

#打补丁
#如果不打补丁的话,下面就必须写上gevent.sleep(0.5)
monkey.patch_all()

def f(n):
    for i in range(n):
        print(gevent.getcurrent(), i)
        #用来模拟一个耗时操作
        time.sleep(1)

gevent.joinall([
    gevent.spawn(f, 5),
    gevent.spawn(f, 5),
    gevent.spawn(f, 5)
])

运行结果:

如果我们不加打补丁,即红色部分,则不会自动切换,运行结果如下:

最后附上一个多协程并发下载示例:

from gevent import monkey
import gevent
import urllib.request

#有IO才做时需要这一句
monkey.patch_all()

def my_downLoad(file_name, url):
    print('GET: %s' % url)
    resp = urllib.request.urlopen(url)
    data = resp.read()

    with open(file_name, "wb") as f:
        f.write(data)

    print('%d bytes received from %s.' % (len(data), url))

gevent.joinall([
        gevent.spawn(my_downLoad, "1.mp4", 'http://....mp4'),
        gevent.spawn(my_downLoad, "2.mp4", 'http://....mp4'),
])

 

转载于:https://www.cnblogs.com/DarrenChan/p/10349142.html

### 回答1: 协程(Coroutine)是一种轻量级的线程,也被称为用户级线程,由用户控制调度,而不是由操作系统控制。在 Python 中,协程是通过生成器实现的。 协程的特点是可以在执行过程中暂停并保存当前状态,然后在需要的时候恢复执行。这种特性使得协程在处理大量并发任务的时候非常高效。协程可以避免线程切换的开销,因为协程在同一个线程内执行,所以不需要进行线程切换,可以利用 CPU 的时间片进行高效的并发处理。 在 Python 中,协程可以使用 asyncio 库实现。通过定义异步函数,可以将阻塞式的 I/O 操作转化为非阻塞式,从而实现高效的并发处理。在 asyncio 中,可以使用 await 关键字来挂起协程,等待异步操作的完成,然后再恢复执行。 总之,协程是一种非常高效的并发处理方式,可以提高程序的性能和响应速度。 ### 回答2: Python中的协程是一种轻量级的并发编程方式。它是一种特殊的函数或方法,可以在函数执行过程中暂停,并且在稍后的某个时候继续执行,而不是一次性地运行完整个函数。 与传统的多线程或多进程编程不同,协程是单线程下的并发编程方式,它可以在一个线程中实现多个任务的并行执行。这种并发模型的好处在于减少了线程切换的开销,并且不需要使用锁来进行同步操作,避免了线程安全问题和死锁等常见的多线程编程难题。 在Python中,可以使用asyncio库来实现协程。通过使用async和await关键字,我们可以将普通的函数变成协程函数,并在其中使用异步操作。协程函数可以包含多个异步操作,这些操作可以是IO操作、网络请求、数据库查询等。 协程的特点是可以暂时挂起当前的任务,并切换到其他任务执行,待到其他任务完成后再切换回来继续执行。这种切换是由协程框架自动完成的,开发者无需关心具体的切换逻辑。这样可以大大提高程序的执行效率,特别适合处理IO密集型的任务。 总结来说,Python中的协程是一种并发编程方式,可以在一个线程中实现多个任务的并发执行。它的特点是轻量、高效,能够提高程序的执行效率,特别适合处理IO密集型的任务。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值