Python 异步编程是指通过非阻塞的方式来执行多个任务的编程方式,这可以大大提高程序的效率,尤其是在处理 I/O 密集型任务时。Python 中的异步编程主要依赖于 asyncio 模块,并且通过 async 和 await 关键字来实现。下面是 Python 异步编程的一些要点:
1. 使用 async def
定义异步函数
def hello():
return 'hello'
print(hello) # <function hello at 0x100ce8e00>
这是一个普通的函数。
async def hello():
return 'hello'
print(hello) # <function hello at 0x102b58e00>
这是一个使用 async def
关键字定义的异步函数。注意,当我们打印它时,它仍然是一个函数类型。
2. 调用异步函数会返回一个协程
def hello():
return 'hello'
print(hello()) # hello
这是一个普通的函数调用,直接返回字符串 hello
。
async def hello():
return 'hello'
print(hello())
# <coroutine object hello at 0x10276f320>
# RuntimeWarning: coroutine 'hello' was never awaited
当我们像调用普通函数一样调用异步函数时,它会返回一个协程对象,而不是它的返回值。同时,我们还会收到一个运行时警告:coroutine 'hello' was never awaited
—— 协程通常需要使用 await
来等待(稍后会详细说明)。
3. 协程的含义
协程是一种特殊的函数,它可以在执行其他任务时被暂停和恢复。协程还可以暂时将控制权交给其他协程。这使得我们能够在同一时间并发执行多个任务。
4. 使用 asyncio.run()
直接运行协程
async def hello():
print('running hello coroutine')
return 'hello'
asyncio.run(hello()) # running hello coroutine
这才是运行协程的正确方式。
需要注意的是,asyncio
是 Python 标准库的一部分,这意味着它已经随着 Python 一起安装好了,我们不需要安装任何额外的第三方库。我们只需要 import asyncio
,就可以开始使用它了。
asyncio.run() 用于启动一个事件循环,并在循环中执行传入的协程。
5. 使用 await
运行协程
await 用于暂停协程的执行,直到另一个协程完成。在等待时,事件循环可以切换到其他协程,提升了程序的并发能力。
假设我们有一个 hello
协程和一个 main
协程:
import asyncio
async def hello():
print('running hello coroutine')
return 'hello'
async def main():
x = await hello()
print(x)
asyncio.run(main())
# running hello coroutine
# hello
如果我们在另一个协程 main
中调用 hello
,就需要使用 await
关键字。可以将 await hello()
理解为“等待 hello()
完成,然后将返回值赋给 x
”,这就是为什么我们打印 x
时会得到 hello
。
- await 后面可以跟协程、任务、Future 对象等。
- await 是非阻塞的,执行时不会阻塞整个程序的运行。
6. 只能在使用 async def
定义的函数中使用 await
import asyncio
async def hello():
print('running hello coroutine')
return 'hello'
def test():
x = await hello()
test()
# SyntaxError: 'await' outside async function
在这里,我们试图在普通函数 test
中使用 await
,因此会引发语法错误。如果要使用 await
关键字,它必须在使用 async def
定义的异步函数中。
7. 使用 asyncio.create_task
封装任务
任务是协程的封装,它表示一个正在运行的协程,并提供了协程执行的状态信息。我们可以通过 asyncio.create_task() 或 loop.create_task() 来将协程包装成任务。
import asyncio
async def my_coroutine():
print("开始执行")
await asyncio.sleep(1)
print("执行完成")
# 创建并运行多个任务
async def main():
task1 = asyncio.create_task(my_coroutine())
task2 = asyncio.create_task(my_coroutine())
await task1
await task2
asyncio.run(main())
asyncio.create_task() 将协程包装成任务并开始执行,await 用于等待任务完成。
8. 使用 asyncio.gather
并发运行多个协程
import asyncio
async def hello():
print('start')
await asyncio.sleep(1)
print('end')
async def main():
await asyncio.gather(hello(), hello(), hello())
asyncio.run(main())
# start
# start
# start
# end
# end
# end
运行这个脚本时,会首先打印出三个 start
,然后大约 1 秒后打印出三个 end
。
发生了什么?
asyncio.sleep(1)
使协程暂停 1 秒。asyncio.gather
并发运行三个hello()
协程。- 这就是为什么所有的
start
都一起打印出来,所有的end
也一起打印出来。
总结
Python 的异步编程得益于 asyncio 库,它为 Python 提供了原生的协程支持,让编写并发任务变得更加高效和简洁。asyncio 是 Python 3.4 引入的,随着 Python 3.5+ 的 async 和 await 语法的加入,asyncio 的异步编程模型变得更加直观和易用。
asyncio 是一个基于 事件循环 的库,它允许你通过 协程 来处理异步 I/O 操作。其核心思想是通过事件循环来调度任务执行,使得 I/O 密集型操作(如文件读写、网络请求)不会阻塞整个程序的执行。通过这种方式,Python 能够在单线程内高效地执行多个任务,避免了传统多线程编程中线程切换的开销。
本文由mdnice多平台发布
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。