我一直在搞乱PyQt和跨线程的信号/插槽。 在这种情况下,我无法找到我的错误:
我有一个类( MultipleProcessLauncher ),它能够在不同的线程中启动多个进程。 我捕获每个进程的stdout并将这些消息发送到另一个线程( OutputWorker )读取的单个队列,最后一个线程应该发送一个onNewMessage信号(我认为它没有)捕获主类,但永远不会调用回调函数。
print(item)
打印它们)但是: - 读取线程的信号似乎没有发出任何内容,因此主线程的回调函数永远不会被调用...
我非常感谢你的帮助,我想我错过了一些有线程信号的东西......
class OutputWorker(QObject):
onNewMessage = pyqtSignal(['QString'])
def __init__(self, queue, parent=None):
super(OutputWorker, self).__init__(parent)
self.queue = queue
def work(self):
while True:
item = self.queue.get()
self.onNewMessage.emit(item)
self.queue.task_done()
class MultipleProcessLauncher(QObject):
commandEvent = pyqtSignal(['QString'])
def __init__(self, parent=None):
super(MultipleProcessLauncher, self).__init__(parent)
self.messaging_queue = Queue()
# Start reading message
self.reading_thread = QThread()
self.worker = OutputWorker(self.messaging_queue)
self.worker.moveToThread(self.reading_thread)
self.worker.onNewMessage.connect(self.command_event)
self.reading_thread.started.connect(self.worker.work)
self.reading_thread.start()
def execute(self, command):
p = subprocess.Popen(command, stdout=subprocess.PIPE)
t = Thread(target=self.enqueue, args=(p.stdout, self.messaging_queue))
t.daemon = True
t.start()
def enqueue(self, stdout, queue):
for line in iter(stdout.readline, b''):
queue.put(line.decode())
stdout.close()
def command_event(self, event):
# This point is never reached
print('message received')
if __name__ == '__main__':
manager = MultipleProcessLauncher()
manager.execute('ipconfig')
time.sleep(100)
答案 0 :(得分:3)
Qt的跨线程信令基于事件循环,因此您需要执行QApplication,以便有一个主事件循环来处理来自其他线程的信号。例如:
if __name__ == '__main__':
app = QApplication([])
manager = MultipleProcessLauncher()
manager.execute('ipconfig')
MAX_WAIT_MSEC = 100 * 1000 # 100 seconds
QTimer.singleShot(MAX_WAIT_MSEC, app.quit)
app.exec()
在您的实际应用程序中,您可能会根据用户输入执行管理器,因此执行将在一个插槽中,并且不需要退出等等,但您明白了。