这是由于asyncio.gather在等待所有任务完成时会阻塞事件循环,直到所有任务都完成后才能恢复事件循环。而在从STDIN请求输入时,事件循环在等待输入时也会被阻塞。因此,任务不会被执行直到输入完成。
为了解决这个问题,我们可以使用asyncio.wait()代替asyncio.gather()。这样,事件循环在等待结果时不会被阻塞,任务会被并发执行。
下面是示例代码,演示了如何使用asyncio.wait()来并发执行任务:
import asyncio
async def task1():
print("Task 1 started")
await asyncio.sleep(2)
print("Task 1 completed")
async def task2():
print("Task 2 started")
await asyncio.sleep(1)
print("Task 2 completed")
async def user_input():
print("Enter something: ")
result = await asyncio.get_event_loop().run_in_executor(None, input)
print(f"You entered {result}")
async def main():
tasks = [task1(), task2(), user_input()]
done, _ = await asyncio.wait(tasks, return_when=asyncio.FIRST_COMPLETED)
# Print any results from completed tasks
for task in done:
print(task.result())
asyncio.run(main())
在这个例子中,我们创建了三个任务,两个休眠2秒和1秒,另一个等待用户输入。我们使用asyncio.wait()代替asyncio.gather()来并发执行这些任务。我们还使用了asyncio.run_in_executor()在子线程中运行input()函数,以避免阻塞事件循环。
我们将return_when参数设置为asyncio.FIRST_COMPLETED以在至少有一个任务完成时继续事件循环。最后,我们遍历已完成的任务,并打印任何结果。