0

0

在 Python 中无需等待即可启动或恢复异步方法/协程

心靈之曲

心靈之曲

发布时间:2025-10-18 11:40:01

|

982人浏览过

|

来源于php中文网

原创

 在 Python 中无需等待即可启动或恢复异步方法/协程

本文旨在解决在 python 中启动异步协程时遇到的困惑,并提供一种在不阻塞主线程的情况下,类似 javascript 的方式立即执行异步任务的方案。文章深入探讨了 `asyncio` 库的特性,并结合 `run_coroutine_threadsafe` 方法展示了如何在独立的事件循环中运行协程,从而实现异步任务的并行执行和状态监控。

在 Python 的异步编程中,一个常见的困惑是如何立即启动一个异步协程,而无需立即 `await` 它。与 JavaScript 等语言不同,Python 的 `async` 函数在调用时并不会立即执行,而是返回一个协程对象。只有通过 `await`、`asyncio.gather` 等方式,协程才会真正开始执行。 这种行为可能导致一些问题,尤其是在需要尽快启动多个协程,并在稍后等待它们完成的情况下。如果使用 `asyncio.gather`,所有协程会在同一时刻启动,导致潜在的性能瓶颈。 为了解决这个问题,我们可以利用 `asyncio.run_coroutine_threadsafe` 函数,在一个独立的线程中运行协程。这允许协程立即启动,而不会阻塞主线程。 **使用 `run_coroutine_threadsafe`** `asyncio.run_coroutine_threadsafe(coro, loop)` 函数接受一个协程对象 `coro` 和一个事件循环 `loop` 作为参数。它将协程提交到指定的事件循环中,并在该循环中运行协程。该函数返回一个 `concurrent.futures.Future` 对象,可以用来获取协程的结果或检查其状态。 以下是一个示例代码,展示了如何使用 `run_coroutine_threadsafe` 函数: ```python import asyncio import time from threading import Thread global_loop = None def thread_for_event_loop(): global global_loop global_loop = asyncio.new_event_loop() asyncio.set_event_loop(global_loop) global_loop.run_forever() t = Thread(target=thread_for_event_loop) t.daemon = True t.start() time.sleep(1) # wait for thread to start old_print = print print = lambda *_: old_print(round(time.perf_counter(), 1), *_) def attempt(future): # doesn't actually do anything, only prints if task is done print(future.done()) async def work(): print("SETUP") await asyncio.sleep(2) print("MIDDLE") await asyncio.sleep(2) print("END") return "Result" async def main(): print("START", int(time.perf_counter())) task = asyncio.run_coroutine_threadsafe(work(), global_loop) attempt(task) attempt(task) print("before first sleep") time.sleep(3) print("after first sleep") attempt(task) attempt(task) print("before second sleep") time.sleep(3) # Block CPU to wait for second sleeping to finish print("after second sleep") attempt(task) attempt(task) print(await asyncio.wrap_future(task)) asyncio.run(main())

代码解释:

  1. 创建事件循环线程: thread_for_event_loop 函数创建一个新的事件循环,并在一个独立的线程中运行它。这确保了协程在后台运行,而不会阻塞主线程。
  2. attempt 函数: attempt 函数用于检查协程的状态。它接受一个 Future 对象作为参数,并打印协程是否完成。
  3. work 协程: work 协程模拟一个耗时任务,其中包含多个 await asyncio.sleep() 调用。
  4. main 协程: main 协程使用 asyncio.run_coroutine_threadsafe 函数将 work 协程提交到事件循环线程中。然后,它使用 attempt 函数检查协程的状态,并使用 time.sleep() 模拟主线程中的其他工作。
  5. asyncio.wrap_future: 将 concurrent.futures.Future 包装成 asyncio 的 Future,以便使用 await 等待结果。

运行结果:

1.1 START 1
1.1 False
1.1 False
1.1 before first sleep
1.1 SETUP
3.1 MIDDLE
4.1 after first sleep
4.1 False
4.1 False
4.1 before second sleep
5.1 END
7.1 after second sleep
7.1 True
7.1 True
7.1 Result

注意事项:

  • asyncio.run_coroutine_threadsafe 函数需要一个事件循环作为参数。如果当前线程没有事件循环,则需要创建一个新的事件循环。
  • asyncio.run_coroutine_threadsafe 函数返回一个 concurrent.futures.Future 对象。可以使用 await asyncio.wrap_future(future) 等待 Future 对象完成并获取结果。
  • 在多线程环境中使用 asyncio 时,需要注意线程安全问题。确保所有对 asyncio 对象的访问都是线程安全的。

总结:

PageGen
PageGen

AI页面生成器,支持通过文本、图像、文件和URL一键生成网页。

下载

立即学习Python免费学习笔记(深入)”;

通过使用 asyncio.run_coroutine_threadsafe 函数,我们可以实现类似 JavaScript 的异步编程模型,即立即启动异步任务,并在稍后等待它们完成。这种方法可以提高程序的性能和响应速度,尤其是在需要同时运行多个异步任务的情况下。

					

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
python中print函数的用法
python中print函数的用法

python中print函数的语法是“print(value1, value2, ..., sep=' ', end=' ', file=sys.stdout, flush=False)”。本专题为大家提供print相关的文章、下载、课程内容,供大家免费下载体验。

186

2023.09.27

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

776

2023.08.22

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

443

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

544

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

73

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

207

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

191

2025.11.08

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

84

2026.01.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 4.2万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 2.5万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号