关于python并发编程中的协程

作者:菜鸟小超 时间:2023-10-18 04:37:44 

什么是协程

协程(Coroutine)是一种比线程更加轻量级的并发方式,它不需要线程上下文切换的开销,可以在单线程中实现并发。协程通常具有以下特点:

  • 协程中的代码可以暂停执行,并且在需要的时候可以恢复执行。

  • 多个协程可以在同一线程中并发执行,但是任意时刻只有一个协程在执行。

  • 协程通常是基于事件循环(Event Loop)实现的,事件循环负责调度协程的执行。

协程和线程

线程和协程都是实现并发编程的方式,但它们有一些不同的特点和应用场景。

**线程是操作系统调度的基本单位,**每个线程都拥有自己的执行上下文,包括线程栈、寄存器等。线程之间的切换需要进行上下文切换,包括保存当前线程的上下文,恢复另一个线程的上下文等操作,这些操作会耗费大量的时间和资源。在多线程编程中,线程切换是非常常见的操作,原因如下:

  1. 调度。当多个线程同时执行时,操作系统需要对这些线程进行调度,根据优先级等因素决定当前应该执行哪个线程。线程切换是调度的基本操作之一,通过切换线程,操作系统可以实现多个线程的并发执行。

  2. 等待。当一个线程需要等待某个事件发生时,例如等待 IO 操作完成、等待锁释放等,线程可以主动释放 CPU,使其他线程有机会执行。在等待完成后,线程可以被重新唤醒,继续执行。

  3. 并发。线程可以实现并发执行的效果,例如一个线程处理网络请求,另一个线程处理用户交互,这样可以提高系统的响应速度和处理能力。

  4. 切换到其他线程执行。在某些情况下,线程可能会因为一些原因无法继续执行,例如线程进入了死循环或者发生了异常,这时需要切换到其他线程执行,避免系统崩溃或者出现其他问题。

线程的并发编程通常会受到多线程竞争、死锁、上下文切换等问题的限制。在 Python 中,使用多线程编程需要注意线程安全、GIL 等问题。

协程是一种轻量级的并发方式,它是在用户空间中实现的,并不依赖于操作系统的调度。协程可以在同一个线程中实现并发,不需要进行上下文切换,因此执行效率非常高。协程通常使用事件循环(Event Loop)来调度协程的执行,事件循环会在协程需要等待 IO 操作或者其他协程时,暂停当前协程的执行,执行其他协程,从而实现并发执行的效果。在 Python 中,协程通常使用 asyncio 模块来实现,支持异步 IO、网络编程、任务调度等场景。

相对于线程,协程的主要优点包括:

  • 更加轻量级,占用的资源更少;

  • 不需要进行上下文切换,执行效率更高;

  • 可以使用事件循环进行调度,实现高并发的效果;

  • 不会受到 GIL 的限制,可以更好地利用多核 CPU。

然而,协程也有一些限制,例如无法利用多核 CPU、调试困难等问题。在选择使用线程还是协程时,需要根据具体的应用场景进行选择。

协程的应用

协程可以应用于很多场景,例如:

  • 网络编程:协程可以帮助我们实现高并发的网络应用。

  • 异步IO:协程可以帮助我们高效地处理异步IO操作。

  • 数据库操作:协程可以帮助我们实现高并发的数据库应用。

  • 任务调度:协程可以帮助我们实现高效的任务调度系统。

演示Demo

下面是一个示例代码,演示了如何使用协程和 asyncio 模块来实现一个简单的任务调度:

import asyncio

async def task1():
   print("Task 1")
   await asyncio.sleep(1)
   print("Task 1 done")

async def task2():
   print("Task 2")
   await asyncio.sleep(2)
   print("Task 2 done")

async def task3():
   print("Task 3")
   await asyncio.sleep(3)
   print("Task 3 done")

async def main():
   await asyncio.gather(task1(), task2(), task3())

这段代码使用了 Python 的协程和 asyncio 模块,定义了三个协程函数 task1task2task3,以及一个主协程函数 main。每个协程函数打印自己的任务名,然后暂停一段时间。主协程函数使用 asyncio.gather 并发执行了三个协程函数,最终输出结果为:

Task 1
Task 2
Task 3
Task 1 done
Task 2 done
Task 3 done
[Finished in 3.2s]

来源:https://blog.csdn.net/weixin_41777118/article/details/130277124

标签:python,并发编程,协程
0
投稿

猜你喜欢

  • python 文件下载之断点续传的实现

    2023-07-18 21:57:16
  • Python 弹窗设计小人发射爱心

    2021-07-16 14:41:01
  • Python实现Logger打印功能的方法详解

    2023-04-16 23:02:43
  • EXECUTE IMMEDIATE用法小结

    2009-09-26 18:32:00
  • 解析ASP与SQL server互操作的时间处理

    2008-05-17 11:57:00
  • Python列表推导式实现代码实例

    2023-02-07 21:35:05
  • 浅谈flask中的before_request与after_request

    2021-05-10 01:29:11
  • python ElementTree 基本读操作示例

    2022-10-23 07:27:25
  • ASP经常用到的函数

    2009-07-06 13:00:00
  • SQL 重复记录问题的处理方法小结

    2024-01-16 14:56:36
  • 使用go xorm来操作mysql的方法实例

    2024-01-18 11:56:35
  • vue watch监控对象的简单方法示例

    2024-05-05 09:11:00
  • 把pandas转换int型为str型的方法

    2022-02-16 15:45:03
  • 解决numpy数组互换两行及赋值的问题

    2023-07-26 16:51:24
  • Windows下安装python MySQLdb遇到的问题及解决方法

    2022-07-20 13:22:36
  • Go json反序列化“null“的问题解决

    2024-02-18 22:54:19
  • JavaScript的事件机制详解

    2024-05-03 15:58:45
  • "模板化"——限制还是激发

    2009-03-26 11:36:00
  • python如何编写win程序

    2022-12-09 11:48:38
  • ASP.NET MVC Bundles 用法和说明(打包javascript和css)

    2023-07-17 01:20:59
  • asp之家 网络编程 m.aspxhome.com