Получение значений из функций, выполняемых как задачи asyncio
Я пробовал следующий код:
import asyncio
@asyncio.coroutine
def func_normal():
print("A")
yield from asyncio.sleep(5)
print("B")
return 'saad'
@asyncio.coroutine
def func_infinite():
i = 0
while i<10:
print("--"+str(i))
i = i+1
return('saad2')
loop = asyncio.get_event_loop()
tasks = [
asyncio.async(func_normal()),
asyncio.async(func_infinite())]
loop.run_until_complete(asyncio.wait(tasks))
loop.close()
Я не могу понять, как получить значения в переменных из этих функций. Я не могу этого сделать:
asyncio.async(a = func_infinite())
поскольку это сделало бы это аргументом ключевого слова. Как я могу это сделать?
Ответы
Ответ 1
Работы работают так, как есть. Просто используйте возвращаемое значение из loop.run_until_complete()
и call asyncio.gather()
для сбора нескольких результатов:
#!/usr/bin/env python3
import asyncio
@asyncio.coroutine
def func_normal():
print('A')
yield from asyncio.sleep(5)
print('B')
return 'saad'
@asyncio.coroutine
def func_infinite():
for i in range(10):
print("--%d" % i)
return 'saad2'
loop = asyncio.get_event_loop()
tasks = func_normal(), func_infinite()
a, b = loop.run_until_complete(asyncio.gather(*tasks))
print("func_normal()={a}, func_infinite()={b}".format(**vars()))
loop.close()
Выход
--0
--1
--2
--3
--4
--5
--6
--7
--8
--9
A
B
func_normal()=saad, func_infinite()=saad2
Ответ 2
loop.run_until_complete
возвращает значение, возвращаемое функцией, которую вы передаете. Таким образом, он вернет вывод asyncio.wait
:
import asyncio
@asyncio.coroutine
def func_normal():
print("A")
yield from asyncio.sleep(5)
print("B")
return 'saad'
@asyncio.coroutine
def func_infinite():
i = 0
while i<10:
print("--"+str(i))
i = i+1
return('saad2')
loop = asyncio.get_event_loop()
tasks = [
asyncio.async(func_normal()),
asyncio.async(func_infinite())]
done, _ = loop.run_until_complete(asyncio.wait(tasks))
for fut in done:
print("return value is {}".format(fut.result()))
loop.close()
Вывод:
A
--0
--1
--2
--3
--4
--5
--6
--7
--8
--9
B
return value is saad2
return value is saad
Вы также можете получить доступ к результатам непосредственно из массива tasks
:
print(tasks[0].result())
print(tasks[1].result())
Ответ 3
Если вы хотите использовать какое-либо значение, возвращаемое сопрограммой coroutine, как только закончите coroutine, вы можете передать будущий объект в coro и обновить это будущее по вычисленному значению. Как только будущее будет обновлено, он передаст свой future.result() функции обратного вызова, которая связана с данным будущим. См. Следующий код:
import asyncio
async def func_normal(future):
print("A")
await asyncio.sleep(5)
print("B")
# return 'saad'
future.set_result('saad')
async def func_infinite(future):
i = 0
while i<10:
print("--"+str(i))
i = i+1
# return('saad2')
future.set_result('saad2')
def got_result(future):
print(future.result())
loop = asyncio.get_event_loop()
future1 = asyncio.Future()
future2 = asyncio.Future()
future1.add_done_callback(got_result)
future2.add_done_callback(got_result)
# Coros are automatically wrapped in Tasks by asyncio.wait()
coros = [
func_normal(future1),
func_infinite(future2)]
loop.run_until_complete(asyncio.wait(coros))
loop.close()
Функция обратного вызова вызывается с единственным аргументом - будущим объектом, с которым он связан. Если вам нужно передать больше аргументов в обратный вызов, используйте частичный пакет functools:
future1.add_done_callback(functools.partial(print, "future:", argin))
вызовет
print("future:", argin)