我准备在PyQt程序中嵌入一个iohttp服务器,但是当我运行下面的代码时,Qt窗口无法显示,我知道这是由于以下原因造成的。web.run_app(app)
我曾试图把它移到一个线程,但后来我得到了 RuntimeError: There is no current event loop in thread 'Dummy-1'
那我该怎么做?我发现 asyncqt 这可能会有帮助,但我不知道如何用它来处理aiohttp服务器。
from PyQt5.QtCore import *
from PyQt5.QtGui import *
from PyQt5.QtWidgets import *
from aiohttp import web
class ThreadGo(QThread): # threading.Thread
# implementing new slots in a QThread subclass is error-prone and discouraged.
def __init__(self, parent, func, *args, **kwargs):
super().__init__(parent)
self.func = func
self.args = args
self.kwargs = kwargs
self.result = 0
onFinished = self.kwargs.get('onFinished')
self.finished.connect(onFinished) if onFinished else None # 用lambda还不行呢
self.finished.connect(self.deleteLater)
self.start()
def run(self):
self.result = self.func(*self.args) # deleteLater
class Window(QMainWindow):
def __init__(self, parent=None, **kwargs):
super().__init__(parent, **kwargs)
self.setUpHTTPServer()
def setUpHTTPServer(self):
async def hello(request):
return web.Response(text="Hello, world")
app = web.Application()
app.add_routes([web.get('/', hello)])
web.run_app(app)
# ThreadGo(self, lambda:web.run_app(app))#get RuntimeError: There is no current event loop in thread 'Dummy-1
if __name__ == "__main__":
from sys import argv, exit
a = QApplication(argv)
w = Window()
w.show()
exit(a.exec_())
在下面的例子中,我展示了如何使用Qt与aiohttp服务器。
import asyncio
from functools import cached_property
from PyQt5.QtWidgets import QApplication, QMainWindow
from asyncqt import QEventLoop
from aiohttp import web
class Window(QMainWindow):
def __init__(self, parent=None, **kwargs):
super().__init__(parent, **kwargs)
self.setup_server()
def setup_server(self):
self.app.add_routes([web.get("/", self.hello)])
@cached_property
def app(self):
return web.Application()
async def hello(self, request):
return web.Response(text="Hello, world")
def run(self):
web.run_app(self.app)
def main():
import sys
a = QApplication(sys.argv)
loop = QEventLoop(a)
asyncio.set_event_loop(loop)
w = Window()
w.show()
w.run()
if __name__ == "__main__":
main()