深入探索 Python 中的 asyncio:异步编程的利器

news2025/3/31 10:01:01

在当今的软件开发中,异步编程已经成为了提高程序性能和响应能力的重要手段之一。Python 作为一种广泛使用的编程语言,提供了强大的异步编程支持,而 asyncio 库则是其中的核心。本文将深入探讨 asyncio 的基本概念、使用方法以及一些高级特性,帮助读者更好地理解和应用异步编程。
一、异步编程简介
在传统的同步编程中,程序的执行是顺序的,一个任务完成后才会开始下一个任务。如果某个任务需要等待(例如 I/O 操作),整个程序就会被阻塞,直到该任务完成。这种模式在处理 I/O 密集型任务时效率低下,因为 CPU 在等待 I/O 的过程中处于空闲状态。
异步编程则允许程序在等待某个任务完成时,切换到其他任务的执行,从而充分利用 CPU 的计算能力。Python 中的 asyncio 库就是为异步编程而设计的,它提供了一种基于事件循环的编程模型,使得程序可以在等待 I/O 操作时执行其他任务。
二、asyncio 的基本概念
1. 事件循环(Event Loop)
事件循环是 asyncio 的核心,它负责调度和管理异步任务。事件循环会不断地检查是否有任务可以执行,如果有,则将其放入执行队列中;如果没有,则等待某个任务完成。事件循环的运行是异步编程的基础。
在 Python 中,可以通过以下代码获取默认的事件循环:

import asyncio

loop = asyncio.get_event_loop()

2. 协程(Coroutine)
协程是异步编程中的基本单元,它是一种特殊的函数,可以通过 async 关键字定义。协程可以被暂停和恢复,这使得它可以在等待 I/O 操作时释放控制权,让其他任务可以执行。
以下是一个简单的协程示例:

async def hello():
    print("Hello")
    await asyncio.sleep(1)
    print("World")

3. 任务(Task)
任务是协程的封装,它将协程与事件循环联系起来。事件循环可以通过任务来调度协程的执行。可以通过以下代码将协程转换为任务:

task = loop.create_task(hello())

或者使用 asyncio.create_task() 函数:

task = asyncio.create_task(hello())

三、asyncio 的基本使用
1. 运行协程
要运行协程,需要将其转换为任务,并将其加入事件循环。以下是一个完整的示例:

import asyncio

async def hello():
    print("Hello")
    await asyncio.sleep(1)
    print("World")

async def main():
    task1 = asyncio.create_task(hello())
    task2 = asyncio.create_task(hello())
    await task1
    await task2

loop = asyncio.get_event_loop()
loop.run_until_complete(main())

在这个例子中,main 函数创建了两个任务 task1 和 task2,并将它们加入事件循环。await task1 和 await task2 表示等待这两个任务完成。
2. 并发执行
asyncio 支持并发执行多个任务。在上面的例子中,task1 和 task2 是并发执行的,它们的执行顺序是不确定的。这是因为事件循环会根据任务的状态来调度它们的执行。
3. 异步 I/O 操作
asyncio 提供了许多异步 I/O 操作的接口,例如 asyncio.open_connection()、asyncio.start_server() 等。这些接口可以用于异步地进行网络通信、文件操作等。
以下是一个异步网络请求的示例:

import asyncio
import aiohttp

async def fetch(session, url):
    async with session.get(url) as response:
        return await response.text()

async def main():
    async with aiohttp.ClientSession() as session:
        html = await fetch(session, 'http://example.com')
        print(html)

loop = asyncio.get_event_loop()
loop.run_until_complete(main())

在这个例子中,fetch 函数使用 aiohttp 库异步地发送 HTTP 请求,并返回响应的内容。main 函数创建了一个 aiohttp.ClientSession 对象,并调用 fetch 函数获取网页内容。
四、asyncio 的高级特性
1. 锁(Lock)
在并发编程中,锁是一种常用的同步机制。asyncio 提供了 asyncio.Lock 类,用于在协程之间同步访问共享资源。
以下是一个使用锁的示例:

import asyncio

async def worker(lock, num):
    async with lock:
        print(f"Worker {num} is working")
        await asyncio.sleep(1)
        print(f"Worker {num} finished")

async def main():
    lock = asyncio.Lock()
    tasks = [asyncio.create_task(worker(lock, i)) for i in range(5)]
    await asyncio.gather(*tasks)

loop = asyncio.get_event_loop()
loop.run_until_complete(main())

在这个例子中,worker 函数使用 async with lock 语句获取锁,只有当锁被释放后,其他协程才能获取锁并执行。
2. 信号量(Semaphore)
信号量是一种比锁更灵活的同步机制,它可以控制同时访问共享资源的协程数量。asyncio 提供了 asyncio.Semaphore 类,用于实现信号量。
以下是一个使用信号量的示例:

import asyncio

async def worker(semaphore, num):
    async with semaphore:
        print(f"Worker {num} is working")
        await asyncio.sleep(1)
        print(f"Worker {num} finished")

async def main():
    semaphore = asyncio.Semaphore(2)
    tasks = [asyncio.create_task(worker(semaphore, i)) for i in range(5)]
    await asyncio.gather(*tasks)

loop = asyncio.get_event_loop()
loop.run_until_complete(main())

在这个例子中,semaphore 的值为 2,表示最多允许 2 个协程同时访问共享资源。当有超过 2 个协程请求访问时,其他协程会被阻塞,直到有协程释放信号量。
3. 条件变量(Condition)
条件变量是一种用于协程之间通信的机制,它允许协程在满足某个条件时进行通知和等待。asyncio 提供了 asyncio.Condition 类,用于实现条件变量。
以下是一个使用条件变量的示例:

import asyncio

async def consumer(condition):
    async with condition:
        print("Consumer is waiting")
        await condition.wait()
        print("Consumer is notified")

async def producer(condition):
    async with condition:
        print("Producer is producing")
        await asyncio.sleep(1)
        print("Producer is notifying")
        condition.notify_all()

async def main():
    condition = asyncio.Condition()
    consumer_task = asyncio.create_task(consumer(condition))
    producer_task = asyncio.create_task(producer(condition))
    await consumer_task
    await producer_task

loop = asyncio.get_event_loop()
loop.run_until_complete(main())

在这个例子中,consumer 函数使用 await condition.wait() 语句等待条件变量的通知,producer 函数使用 condition.notify_all() 语句通知所有等待的协程。
五、总结
asyncio 是 Python 中强大的异步编程库,它提供了事件循环、协程、任务等基本概念,以及锁、信号量、条件变量等高级特性。通过使用 asyncio,可以实现高效的异步编程,提高程序的性能和响应能力。希望本文能够帮助读者更好地理解和应用 asyncio,在实际开发中发挥其优势。
----
希望这篇文章对你有帮助!如果你有其他主题或需求,欢迎随时告诉我。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2323263.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

STM32硬件IIC与OLED使用

OLED屏幕介绍 OLED即有机发光管(Organic Light-Emitting Diode,OLED)。OLED显示技术具有自发光、广视角、几乎无穷高的对比度、较低功耗、极高反应速度、可用于绕曲性面板、使用温度范围广、构造及制程简单等有点,被认为是下一代的平面显示屏新兴应用技术 OLED显示…

基于Spring Boot的电动车智能充电服务平台的设计与实现(LW+源码+讲解)

专注于大学生项目实战开发,讲解,毕业答疑辅导,欢迎高校老师/同行前辈交流合作✌。 技术范围:SpringBoot、Vue、SSM、HLMT、小程序、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容:…

十、JavaScript对象

一、对象 创建对象的方法有三种:字面量、new、构造函数。 1.利用字面量创建对象 花括号{}里面包含了表达这个具体事物(对象)的属性和方法 // 1.利用对象字面量创建对象{}// var obj {}; // 创建了一个空的对象var obj {uname: black,ag…

FFmpeg开发学习:音视频封装

1.基本流程 1.输入参数 输出文件路径 char *output 视频编码参数 AVCodecParameters *video_par 音频编码参数 AVCodecParameters *audio_par 数据包 AVPacket *packets[] 2.封装流程 (1)创建输出的上下文AVFormatContext指针 AVFormatContext *out_fm…

hackmyvm-reversteg

arp-scan -l nmap -sS -v 192.168.222.45 在源码中可以看到 根据下面的提示可以猜测117db0148dc179a2c2245c5a30e63ab0是一个图像文件 将图片下载到本地 隐写术 在两张图片上使用strings,发现有一些可打印的字符串 strings 117db0148dc179a2c2245c5a30e63ab0.jpg base64解码…

UE4学习笔记 FPS游戏制作17 让机器人持枪 销毁机器人时也销毁机器人的枪 让机器人射击

添加武器插槽 打开机器人的Idle动画,方便查看武器位置 在动画面板里打开骨骼树,找到右手的武器节点,右键添加一个插槽,重命名为RightWeapon,右键插槽,添加一个预览资产,选择Rifle,根…

设计模式(创建型)-建造者模式

定义 建造者模式(Builder Pattern)是一种创建型设计模式,它将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。该模式允许通过多个简单的步骤逐步构建出一个复杂的对象,用户只需指定复杂对象…

Git 之配置ssh

1、打开 Git Bash 终端 2、设置用户名 git config --global user.name tom3、生成公钥 ssh-keygen -t rsa4、查看公钥 cat ~/.ssh/id_rsa.pub5、将查看到的公钥添加到不同Git平台 6、验证ssh远程连接git仓库 ssh -T gitgitee.com ssh -T gitcodeup.aliyun.com

黑马点评项目

遇到问题: 登录流程 session->JWT->SpringSession->tokenRedis (不需要改进为SpringSession,token更广泛,移动端或者前后端分离都可以用) SpringSession配置为redis模式后,redis相当于分布式se…

【AVRCP】AVRCP核心术语解析

目录 一、协议核心术语:架构的基石 1.1 音视频控制协议簇(AVRCP 生态链) 1.2 数据传输协议(L2CAP 核心术语) 二、设备架构术语:角色与交互 2.1 设备角色模型(CT/TG 二元架构) …

【弹性计算】异构计算云服务和 AI 加速器(四):FPGA 虚拟化技术

异构计算云服务和 AI 加速器(四):FPGA 虚拟化技术 🚀 FPGA(Field-Programmable Gate Array,现场可编程门阵列)是一种可重构的半导体芯片,允许用户根据需要动态配置硬件逻辑&#xff…

编译原理——自底向上语法优先分析

文章目录 自底向上优先分析概述一、自底向上优先分析概述二、简单优先分析法(一)优先关系定义(二)简单优先文法的定义(三)简单优先分析法的操作步骤 三、算法优先分析法(一)直观算符…

nuxt3网站文章分享微信 ,QQ功能

1.安装 npm install qrcode --save-dev 2.组件使用 <div class"share"><div style"line-height: 69px; color: #fff;width: 100px;"><p style"text-align: center;">分享:</p></div><div click"shareToMi…

STM32F103_LL库+寄存器学习笔记07 - 串口接收缓冲区非空中断

导言 上一章节《STM32F103_LL库寄存器学习笔记06 - 梳理串口与串行发送“Hello,World"》梳理完USART的基本设置与发送字符串“Hello,World"&#xff0c;接着梳理接收缓冲区非空中断。 实用的串口接收程序都会使用中断方式&#xff0c;不会使用轮询方式。最主要的原因…

生物中心论

Robert Lanza的“生物中心论”&#xff08;Biocentrism&#xff09;是一种以生命和意识为核心的宇宙观&#xff0c;试图颠覆传统科学对时间、空间和物质的理解。 一、核心观点 意识创造宇宙 生物中心论认为&#xff0c;宇宙的存在依赖于观察者的意识。传统科学将宇宙视为独立实…

Spring AOP:面向切面编程的探索之旅

目录 1. AOP 2. Spring AOP 快速入门 2.1 引入 Spring AOP 依赖 2.2 Spring AOP 简单使用 3. Spring AOP 核心概念 3.1 切点 3.1.1 Pointcut 定义切点 3.1.2 切点表达式 3.1.2.1 execution 表达式 3.1.2.2 annotation 表达式 3.2 连接点 3.3 通知(Advice) 3.3.1 通…

使用QT画带有透明效果的图

分辨率&#xff1a;24X24 最大圆 代码: #include <QApplication> #include <QImage> #include <QPainter>int main(int argc, char *argv[]) {QImage image(QSize(24,24),QImage::Format_ARGB32);image.fill(QColor(0,0,0,0));QPainter paint(&image);…

RocketMQ可视化工具使用 - Dashboard(保姆级教程)

1、github拉取代码&#xff0c;地址&#xff1a; https://github.com/apache/rocketmq-dashboard 2、指定Program arguments&#xff0c;本地启动工程 勾上这个Program arguments&#xff0c;会出现多一个对应的框 写入参数 --server.port1280 --rocketmq.config.namesrvAddr…

用Unity实现UDP客户端同步通信

制作UDPNetMgr网络管理模块 这段代码定义了一个名为UDPNetMgr的 Unity 脚本类&#xff0c;用于管理 UDP 网络通信&#xff0c;它作为单例存在&#xff0c;在Awake方法中创建收发消息的线程&#xff0c;Update方法处理接收到的消息&#xff1b;StartClient方法启动客户端连接&a…

pandoc安装及基础使用

pandoc安装 访问pandoc tags,切换至想要安装的版本&#xff0c;本次安装3.6.4 下载windows版本 下载texlive镜像&#xff0c;将文件转换成pdf需要用到 点开后会进入最近的镜像网站 下载完成后解压iso文件&#xff0c;以管理员身份运行install-tl-windows.bat&#xff…