Use asyncio e Tkinter juntos sem congelar a GUI
Eu quero usarasyncio
em combinação com umtkinter
GUI. Eu sou novo emasyncio
e meu entendimento não é muito detalhado. O exemplo aqui inicia 10 tarefas ao clicar no primeiro botão. A tarefa está apenas simulando o trabalho com umsleep()
por alguns segundos.
O código de exemplo está funcionando bem com Python3.6.4rc1
. Mas o problema é que a GUI está congelada. Quando pressiono o primeiro botão e inicio as 10 tarefas assíncronas, não consigo pressionar o segundo botão na GUI até que todas as tarefas estejam concluídas. A GUI nunca deve congelar - esse é o meu objetivo.
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
from tkinter import *
from tkinter import messagebox
import asyncio
import random
def do_freezed():
""" Button-Event-Handler to see if a button on GUI works. """
messagebox.showinfo(message='Tkinter is reacting.')
def do_tasks():
""" Button-Event-Handler starting the asyncio part. """
loop = asyncio.get_event_loop()
try:
loop.run_until_complete(do_urls())
finally:
loop.close()
async def one_url(url):
""" One task. """
sec = random.randint(1, 15)
await asyncio.sleep(sec)
return 'url: {}\tsec: {}'.format(url, sec)
async def do_urls():
""" Creating and starting 10 tasks. """
tasks = [
one_url(url)
for url in range(10)
]
completed, pending = await asyncio.wait(tasks)
results = [task.result() for task in completed]
print('\n'.join(results))
if __name__ == '__main__':
root = Tk()
buttonT = Button(master=root, text='Asyncio Tasks', command=do_tasks)
buttonT.pack()
buttonX = Button(master=root, text='Freezed???', command=do_freezed)
buttonX.pack()
root.mainloop()
Um problema secundário... é que não consigo executar a tarefa uma segunda vez devido a esse erro.
Exception in Tkinter callback
Traceback (most recent call last):
File "/usr/lib/python3.6/tkinter/__init__.py", line 1699, in __call__
return self.func(*args)
File "./tk_simple.py", line 17, in do_tasks
loop.run_until_complete(do_urls())
File "/usr/lib/python3.6/asyncio/base_events.py", line 443, in run_until_complete
self._check_closed()
File "/usr/lib/python3.6/asyncio/base_events.py", line 357, in _check_closed
raise RuntimeError('Event loop is closed')
RuntimeError: Event loop is closed
MultithreadingO multithreading seria uma solução possível? Apenas dois threads - cada loop tem seu próprio thread?